问题描述
在组合两个我拥有的程序时需要帮助,我似乎无法让它对我有用。没有得到想要的输出。
这是我的问题陈述:
在第三个字符串中合并两个单独的字符串并显示它,第一个字符串保持原样,第二个字符串反转。
示例:
输入:
字符串1:“ Hello”
字符串2:'.dlroW'
输出:
“ Hello World”。
示例结尾。
首先:使用字符串函数。(首选)
现在,我刚开始学习汇编语言,所以我想使用字符串函数来学习汇编语言。
第二:不使用字符串函数。
另一种方法是,如果有人可以帮助组合两个程序,一个用于字符串的连接,另一个用于反转,请注意,我已经编写了两个单独的程序,它们运行良好,没有任何打ic,我似乎无法一起做。我要如何处理此问题,是在连接要反转的字符串之前,然后继续添加第二个字符串。但是我似乎无法正常工作。我已尽我所能。
//Concatenation Code
.model tiny
.data
msg1 db 10,13,"Enter the string 1: $"
cat db 30 DUP('$')
msg2 db 10,"Enter the string 2: $"
msg3 db 10,"Concatenated string is: $"
.code
mov ax,@data
mov ds,ax
lea dx,msg1
mov ah,09h
int 21h
lea si,cat
up: mov ah,01h
int 21h
mov [si],al
inc si
cmp al,0dh
jnz up
lea dx,msg2
mov ah,09h
int 21h
dec si
up1: mov ah,01h
int 21h
mov [si],al
inc si
cmp al,0dh
jnz up1
lea dx,msg3
mov ah,09h
int 21h
lea dx,cat
mov ah,09h
int 21h
mov ah,4ch
int 21h
end`
第2部分
//Reversal Code
.model tiny
.data
msg1 db 10,"enter the string: $"
string db 40 DUP('$')
rev db 40 DUP('$')
msg2 db 10,"reverse string is: $"
.code
mov ax,09h
int 21h
mov ah,0ah
lea dx,string
int 21h
lea si,string
lea di,rev
mov cl,[si+1]
mov ch,00h
add di,cx
inc si
inc si
up: mov al,[si]
mov [di],al
inc si
dec di
loop up
inc di
mov ah,09h
lea dx,msg2
int 21h
mov ah,[di]
int 21h
mov ah,4ch
int 21h
end
这是我将两者结合起来的代码。
//That's the code I tried Combining
.model tiny
.data
.model tiny
.data
msg1 db 10,"Enter string1: $"
cat db 30 DUP('$')
msg2 db 10,"Enter string2: $"
msg3 db 10,cat
up: mov ah,01h
int 21h
mov [si],al
inc si
cmp al,0dh
jnz up
lea dx,msg2
mov ah,09h
int 21h
dec si
up2:mov al,[si]
mov [di],al
inc si
dec di
loop up2
inc di
up1:mov ah,0dh
jnz up1
lea dx,msg3
mov ah,09h
int 21h
lea dx,cat
mov ah,4ch
int 21h
end
您可以清楚地看到,我没有正确执行任何一项任务。有人可以告诉我我要去哪里错吗?还是教我如何使用“函数”字符串来做到这一点?
解决方法
尝试进行字符串反转的 up2 循环来得太早了!。您已将其放置在尚未输入第二个字符串(需要反转的字符串)的地方。
如果您在程序中写过注释,那么您自己可能会注意到这一点。
此 up2 循环使用取决于LOOP
寄存器的CX
指令,但是您的程序没有为CX
分配任何合适的值。
您的反向程序也使用2个缓冲区。为什么然后您期望组合从单个缓冲区开始工作?
定义 cat 缓冲区,使其可以容纳两个字符串。
定义 str 缓冲区,使其可以容纳第二个字符串。
lea dx,msg1
mov ah,09h ; DOS.PrintString
int 21h
lea di,cat
up: ; Input f i r s t string
mov ah,01h ; DOS.GetCharacter
int 21h ; -> AL
mov [di],al
inc di
cmp al,13
jne up
dec di ; Throw out the 13
; This marks the start of the reversed string,VERY IMPORTANT
; So don't change DI while inputting the 2nd string
lea dx,msg2
mov ah,09h ; DOS.PrintString
int 21h
lea si,str
mov dx,si
up1: ; Input s e c o n d string
mov ah,01h ; DOS.GetCharacter
int 21h ; -> AL
mov [si],al
inc si
cmp al,13
jne up1
dec si ; Throw out the 13
cmp si,dx
je done ; Second string was empty. CAN HAPPEN!
up2: ; Reversed copying of s e c o n d string
dec si
mov al,[si]
mov [di],al
inc di
cmp si,dx
ja up2
done:
mov ax,0A0Dh ; Add a proper carriage return and linefeed to the result
mov [di],ax
mov al,'$' ; Terminate the result with a dollar sign
mov [di+2],al
lea dx,msg3
mov ah,09h ; DOS.PrintString
int 21h
lea dx,cat
mov ah,09h ; DOS.PrintString
int 21h
首先:使用字符串函数。(首选)
在 up 循环和 up2 循环中,您是否都找到下一条指令:
mov [di],al
inc di
提供
- 方向标记DF很清楚,因此
DI
可以递增 -
ES
段寄存器指向@data
您可以将这2条指令替换为一条STOSB
指令。
这是需要放在程序顶部的内容:
.code
mov ax,@data
mov ds,ax
mov es,ax
cld
如果我们允许自己编写多个std
(设置方向标志)和cld
(清除方向标志)指令的愚蠢序列,我们也可以将mov al,[si]
替换为{{1 }}。必须注意保持有效的lodsb
指针(*)。
SI
在设置方向标记(使用 dec si ; (*)
up2: ; Reversed copying of s e c o n d string
std
lodsb ; Due to STD,SI will decrement
cld
stosb ; Due to CLD,DI will increment
cmp si,dx
jae up2 ; (*)
done:
mov ax,0A0Dh ; Add a proper carriage return and linefeed to the result
stosw
mov al,'$' ; Terminate the result with a dollar sign
stosb
)的代码中,最好以std
指令结尾,以便方向标记处于我们最期望的状态!