2

我正在学习汇编课程,并且我已经写出了大部分程序,我只是在替换单词和显示新字符串时遇到了麻烦。该问题要求一个句子,一个要查找的单词以及一个替换它的单词。程序扫描字符串,替换单词的任何实例,并向您显示新字符串。
示例:“天空是蓝色的。” 查找单词:“sky” 替换它的单词:“ocean” 新字符串:“海洋是蓝色的。”

这是我到目前为止所拥有的:

.586
.MODEL FLAT

INCLUDE io.h            ; header file for input/output

.STACK 4096

.DATA
prompt1 BYTE    "String to Search: ", 0
prompt2 BYTE    "Word to Search For: ", 0
prompt3 BYTE    "Word to replace with: ", 0
target  BYTE    80 DUP (?)
key     BYTE    80 DUP (?)
strSub  BYTE    80 DUP (?)
trgtLength  DWORD   ?
keyLength   DWORD   ?
lastPosn    DWORD   ?
strSubLen   DWORD   ?
resultLbl BYTE  "The new sentence is: ", 0

.CODE
_MainProc PROC
    input prompt1, target, 80   ;input target string
    lea eax, target             ;address of target
    push eax                    ;parameter
    call strlen                 ;strlen(target)
    add esp, 4                  ;remove parameter
    mov trgtLength, eax         ;save length of target
    input prompt2, key, 80      ;input key string
    lea eax, key                ;address of key
    push eax                    ;parameter
    call strlen                 ;strlen(key)
    add esp, 4                  ;remove parameter
    mov keyLength, eax          ;save length of key
    input prompt3, strSub, 80   ;input word to search for
            lea eax, strSub             ;address of key
    push eax                    ;parameter
    call strlen                 ;strlen(strSub)
    add esp, 4                  ;remove parameter
    mov strSubLen, eax          ;save length of key

    mov eax, trgtLength
    sub eax, keyLength
    inc eax                     ;trgtLength - keyLength +1
    mov lastPosn, eax
    cld                         ;Left to Right comparison
    mov eax, 1                  ;starting position

    whilePosn:
        cmp eax, lastPosn       ;position <= last_posn?
        jnle endWhilePosn       ;exit if past last position

        lea esi, target         ;address of target string
        add esi, eax            ;add position
        dec esi                 ;address of position to check
        lea edi, key            ;address of key
        mov ecx, keyLength      ;number of position to check
        repe cmpsb              ;check
        jz found                ;exit of success
        inc eax                 ;increment position
        jmp whilePosn           ;repeat

    endWhilePosn:
        output resultLbl, [esi] ;display new sentence
        jmp quit

    found:
        sub edi, keyLength
        mov ecx, strSubLen
        lea esi, strSub
        cld
        rep movsb
        inc eax
        jmp whilePosn

    quit:
        mov     eax, 0  ; exit with return code 0
        ret
_MainProc ENDP


strlen  PROC
push ebp                    ;establish stack frame
mov ebp, esp
push ebx                    ;save EBX
sub eax, eax                ;length := 0
mov ebx, [ebp+8]            ;address of string

whileChar:
cmp BYTE PTR [ebx], 0       ;null byte?
je endWhileChar             ;exit if so
inc eax                     ;increment length
inc ebx                     ;point at next character
jmp whileChar               ;repeat

endWhileChar:
pop ebx                     ;restore registers
pop ebp
ret
strlen  ENDP
END

该代码可以找到我想要切换的单词,但实际上切换单词是在欺骗我。这本书说目标字符串应该在 EDI 中,要替换的单词应该在 ESI 中,但是他们作为示例给出的代码在 ESI 中具有目标字符串,在 EDI 中替换单词(就像我在这里一样)。

这本书在解释“rep”和“movs”指令方面也做得非常糟糕,所以我有 90% 的把握我的“找到”代码块将是问题所在。任何帮助深表感谢。

4

1 回答 1

2

在 x86 汇编中处理字符串时,必须快速掌握rep指令。如果没有,您可能会在您的第一个程序完成之前去世。

可以在此处(或此处)找到有关汇编中字符串操作的非常好的介绍,但您真正需要的是以下内容:

于 2013-04-24T11:00:57.620 回答