string - 字符串中 2 个字符之间的汇编切换

标签 string assembly x86-16

我正在尝试在字符串中的 2 个字符之间翻转,所以我 函数获取指向 2 个字符的指针,然后切换它们。该函数可以很好地处理常规的汇编字符,但是当我从字符串中将参数作为字符传递时,它开始变得疯狂并从其他地方翻转字符。 例如,我得到这个字符串:

string db 'd', 'a', 'b', 'e', 'c', 0Dh, 0Ah, '$'

我想在“d”和“e”之间切换。 所以我这样做是为了将它们移动到函数中:

lea bx, string
lea ax, [bx] ;i use lea to get the pointer not the char
push ax

lea ax, [bx + 3]
push ax

; i know push 2 parameters.

但由于某种原因,切换后字符串更改为:

string db 'e', 'c', 'b', 'd', 'a', 0Dh, 0Ah, '$'

所以它成功地在两个字符之间交换,但由于某种原因,该函数对它们后面的字符做了同样的事情:

[bx + 1][bx + 4]。 这是为什么?这是我到目前为止的代码:(print = print new line, print_ax_str 这是打印 ax 内的 ptr 字符串的函数。所有来自 inc 文件的内容)

org 100h

jmp main

string db 'd', 'a', 'b', 'e', 'c', 0Dh, 0Ah, '$'
length dw 3

main:
    xor ax, ax
    xor bx, bx
    xor dx, dx
    xor cx, cx
    xor si, si
    xor bp, bp

    lea bx, string
    mov ax, bx
    print "Before string: "
    call print_ax_str

    lea ax, [bx]
    push ax

    lea ax, [bx + 3]
    push ax

    call swap ;swap the chars.

    lea ax, string
    print "After string: "
    call print_ax_str        
    jmp end        
swap:
    push cx
    push bx
    push dx
    push ax
    push bp


    mov bp, sp

    mov ax, [bp + 12]; ax = ptr char 2

    mov cx, [bp + 14]; cx = ptr char 1

    mov bx, ax
    mov ax, [bx] ;ax = char 2

    mov bx, cx
    mov dx, [bx] ;dx = char 1
    mov [bx], ax ;char 1 = char 2 


    mov bx, [bp + 12]
    mov [bx], dx ;char 2 = char 1 
    mov ax, [bx]

    mov sp, bp
    pop bp
    pop ax
    pop dx
    pop bx
    pop cx
    retn 4


end:
    mov ah, 0
    int 16h
    ret

include prints.inc

最佳答案

mov ax, [bx] ;ax = char 2
mov dx, [bx] ;dx = char 1
mov [bx], ax ;char 1 = char 2 
mov [bx], dx ;char 2 = char 1

您正在处理的字符串由字节组成,但您的程序会交换单词。这就是结果不理想的原因。 作为快速修复,请编写:

mov al, [bx] ;AL = char 2
mov dl, [bx] ;DL = char 1
mov [bx], al ;char 1 = char 2 
mov [bx], dl ;char 2 = char 1

如果您有兴趣编写更好的程序,请重新编写 SWAP 过程。

  • 在推送附加寄存器之前设置BP,以便对参数进行寻址可以使用熟悉的偏移量,例如第一个的[bp+4]参数和第二个参数的 [bp+6]
  • 将参数直接移动到地址寄存器(BXSIDI),这样就可以避免频繁移动内容。<

新版本:

swap:
 push bp
 mov  bp, sp

 push ax
 push dx
 push si
 push di

 mov  di, [bp + 4]  ; ptr char 2
 mov  si, [bp + 6]  ; ptr char 1

 mov  dl, [di]      ; DL = char 2
 mov  al, [si]      ; AL = char 1

 mov  [si], dl      ; char 1 = char 2 
 mov  [di], al      ; char 2 = char 1 

 pop  di
 pop  si
 pop  dx
 pop  ax

 pop  bp
 retn 4

mov ax, [bx]
mov sp, bp
pop bp
pop ax

关于这部分代码的一些注释:

  1. 您是否发现 mov ax,[bx] 指令后跟 pop ax 指令毫无用处?
  2. 您不必编写 mov sp, bp 指令,因为堆栈指针仍然指向压入的寄存器。

关于string - 字符串中 2 个字符之间的汇编切换,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47469920/

相关文章:

c++ - STL中使用了哪种字符串匹配算法?

c - 使用 strtol 验证 ANSI C 中的整数输入

c++ - &string 和 &string[0] 的内存地址不同?

assembly - 计算机如何区分数据和指令?

assembly - 从COM抓取流量

javascript - 在 javascript 的正则表达式中传递变量以进行字符串匹配

assembly - 缓冲输入如何工作

assembly - 我们如何仅通过使用它的名称而不使用与之关联的十六进制数来访问 DOS 函数?

assembly - 第二个扇区无法加载到引导文件中

assembly - 如何识别汇编代码中的调用约定