-1

我一直在比较固定字符串和输入字符串。但是很难弄清楚输入的寄存器是什么,是它al还是ah什么。我只是一个初学者,这对程序员来说是一种痛苦的语言,哈哈。请帮助我真的很感激谢谢:)

mov dx, offset temp             ;string input
mov ah, 0ah
int 21h     

mov bl, "a"                     ;condition
cmp al, bl 
jne aw

mov dx, offset msgTrue          ;true
mov ah, 09
int 21h  

aw:                             ;false
mov dx, offset msgFalse
mov ah, 09
int 21h 


ret
    msg  db 10, 13, "           *         ************************************       *****$"
    msg1 db 10, 13, "          ***        *            Ticketing System      *        ***$"
    msg2 db 10, 13, "         *****       ************************************         *$" 
    msg3 db 10, 13, "                          ==========================$" 
    msg4 db 10, 13, "                          =  (a)Land               =$" 
    msg5 db 10, 13, "                          =  (b)Water              =$"
    msg6 db 10, 13, "                          =  (c)Air                =$" 
    msg7 db 10, 13, "                          ==========================$" 
    msg8 db 10, 13, "                           Choose Travel Type: $"
    temp db 2, 0, 3 dup("$") 

    msgTrue db 10, 13, "                           You selected Land$"
    msgFalse db 10, 13, "                           Invalid Input$" 
4

2 回答 2

2

您使用系统调用 0Ah(缓冲输入)http://spike.scu.edu.au/~barry/interrupts.html#ah0a所以您读取的数据在temp(缓冲区)中

0Ah 从 STDIN 读取 n 个字节到缓冲区

mov bx, OFFSET buffer 在 bx 中推送缓冲区的地址(此处为 temp),这是 0Ah 所需要的

要修复读取的字节数,您可以使用例如mov byte [bx],15

另见http://www.fysnet.net/kbbuffio.htm

mov bl, 'a'                     ;condition
cmp al, bl 
jne aw

比较两个 8 位值(字符),请参阅此http://x86.renejeschke.de/html/file_module_x86_id_35.html(AL、AH为 8 位,AX 为 16 位,EAX 为 32 位(扩展 AX))

有关8-/16-/32-/64 位命名约定,请参阅此https://en.wikibooks.org/wiki/X86_Assembly/X86_Architecture#General-purpose_registers_(16-bit_naming_conventions)

这可用于从输入缓冲区temp读取读取的输入字节并进行比较:

val DB 'a'
mov al, val                    ;condition
mov bx, OFFSET temp            ; address of temp in bx: bx is a pointer to first byte in temp now
;alternatively you can use  lea bx, [temp] 
add bx, 2                     ; structure of input buffer (here temp) http://spike.scu.edu.au/~barry/interrupts.html#dosbuf
mov dl, byte [bx]              ; writes the byte [bx] points to in dl - [bx] dereferences bx - the byte command treats bx as a char* - see https://www.cs.uaf.edu/2006/fall/cs301/lecture/10_02_pointer.html
cmp al, dl 
jne aw
于 2016-03-03T08:18:20.003 回答
1

结果在内存中,您要求系统调用将其放入内存中。见拉尔夫的回答。

像这样检查输入:

    mov   dx, offset temp             ; input buffer
    mov   ah, 0ah                     ; buffered input syscall
    int   21h     

    ; select one of two messages to print
    mov   dx, offset msgTrue
    cmp   byte ptr [temp+2], 'a'      ; compare the first byte of user input
    je  .true
    mov   dx, offset msgFalse         ; conditionally skip this instruction
.true:

    mov ah, 09                        ; print string syscall
    int 21h

注意mov ah/int 0x21代码只出现一次,分支只跳过一条指令。您可以cmov在现代 CPU 上执行此操作,但cmov令人讨厌的是没有立即源编码。

请参阅对 Ralf 的答案的评论,以批评不仅使用直接操作数 with 导致的臃肿代码cmp。在使用汇编时常量时,您也可以通过这种方式占用更少的寄存器。


另一个选项(而不是mov dx, offset msgFalse)是add dx, offset msgFalse - msgTrue,它使立即数操作数很小(适合使用 imm8 编码的 -128 .. 127 范围)。但是,它不会在 16 位代码中保存任何代码字节,因为mov dx, imm16它是 3 个字节(每个 dest 寄存器的专用操作码),add dx, imm8也是三个字节(没有专用操作码)。这将节省 32 位代码中的字节,其中地址为 32 位。

于 2016-03-03T10:21:07.740 回答