watermelon 发表于 2018-9-7 19:49:03

学习笔记

这篇笔记是小白学习王爽汇编语言第二版(https://pan.baidu.com/s/1OwSNdDJbRvASlR5VbNZMWg)第八章实验7时候的反思和感想
这个实验书上说是挺有综合性的,前八章的知识点都有用到,我感觉也是的。

首先编译时候报了一个jump out of range的错,我百度发现是loop循环中代码有点太长了,他说loop什么jmp指令是短指令,jmp
这个指令我知道是跳转并且只在当初第1,2章的CS:IP那里看到debug过,但是实际上还是不是很懂。
不过第九章就要学西转移指令原理了,不慌先

所以为了缩短loop循环程序的长度,我就又嵌套了一个循环,再编译就通过了,起码没有语法错误了,然后就是其中一些基础的东西
偏移量一定要搞清楚,尤其是计算结果的时候写字单元时候的偏移量,否则就会造成编译成功,但是结果不正确。

寻址方式每个人喜好不一样?我就特别喜欢这种,但是书上貌似更喜欢.idata的方式

题目和部分运行结果我截了图在https://pan.baidu.com/s/1rltJk6DGmg6yvgs7YAgDDw中

assume cs:codesg, ds:data, ds:table, ss:stack

data segment
        db '1975','1976','1977','1978','1979','1980','1981','1982','1983'
        db '1984','1985','1986','1987','1988','1989','1990','1991','1992'
        db '1993','1994','1995'
        ;以上表示21年的21个字符串

        dd 16,22,382,1356,2390,8000,16000,24486,50065,97479,140417,197514
        dd 345980,590827,803530,1183000,1843000,2759000,3753000,4649000,5937000
        ;以上表示公司21年总收入的21个数据

        dw 3,7,9,13,28,38,130,220,476,778,1001,1442,2258,2793,4037,5635,8226
        dw 11542,14430,15257,17800
        ;以上表示21年公司雇员的21个数据
data ends


;table: ds:
table segment                                                ;此处的地址为data段以后字节数且满16的倍数
        db 21 dup('year summ ne ?? ')
table ends

stack segment                                                ;栈段
        dw 0,0,0,0,0,0,0,0
stack ends


codesg segment
start:
        mov ax,data
        mov ds,ax

        mov ax,stack
        mov ss,ax
        mov sp,16                                ;设置栈和栈顶指针

        mov bx,0                                ;控制data段的偏移量
        mov bp,0                                ;控制公司总收入的偏移量
        mov di,0                                ;控制公司雇员数的偏移量
        mov si,0                                ;控制结果的偏移量

        mov cx,21                                ;设置21次循环

s:       
        push cx                                        ;压栈,记录外循环次数

        mov cx,4
s0:                                                        ;如果一个循环中间代码太长的话会造成jmp越界的情况
                                                        ;所以为了减少循环中代码的长度,写一个内循环s0
        mov ax,0
        mov al,ds:
        mov ds:,al
        inc bx
        inc si
        loop s0
        pop cx                                        ;弹栈

        mov byte ptr ds:,32                        ;写入空格,空格的ascii码是32

;;;;;;这一段要写总收入

        mov ax,0

        inc si
        mov ax,ds:                                        ;dword的低16位
        mov ds:,ax                        ;将dword的低16位传给结果
        add bp,2
        add si,2

        mov dx,0
        mov dx,ds:
        mov ds:,dx                        ;将dword的高16位传给结果

        add bp,2
        add si,2
        mov byte ptr ds:,32                        ;给结果写入空格

;;;;;;这一段要写公司的雇员数
        inc si
        mov ax,0
        mov ax,ds:
        mov word ptr ds:,ax                        ;将公司的雇员数写入结果
        add di,2

        add si,2
        mov byte ptr ds:,32                        ;给结果写入空格


;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;这段代码计算人均收入
        inc si
       
        mov ax,ds:                                        ;被除数低16位写入ax中
        mov dx,ds:                                        ;被除数高16位写入dx中

        div word ptr ds:                        ;除以除数
        mov ds:,ax                                        ;将商存储到结果中


;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

        add si,2
        mov byte ptr ds:,32                        ;给结果写入空格       

        inc si

        loop s                                                                ;循环结束

        mov ax,4c00h
        int 21h
codesg ends

end start



页: [1]
查看完整版本: 学习笔记