任務一
-
任務1-1
對程序task1_1.asm進行匯編、連接,用debug加載、跟蹤調試,基於結果,回答問題。
task1_1.asmassume ds:data, cs:code, ss:stack data segment db 16 dup(0) ; 預留16個字節單元,初始值均為0 data ends stack segment db 16 dup(0) ;預留16個字節單元,初始值均為0 stack ends code segment start: mov ax, data mov ds, ax mov ax, stack mov ss, ax mov sp, 16 ; 設置棧頂 mov ah, 4ch int 21h code ends end start
① 在debug中將執行到line17結束、line19之前,記錄此時:寄存器(DS) =
076A
, 寄存器(SS) =076B
, 寄存器(CS) =076C
② 假設程序加載后,code段的段地址是X,則,data段的段地址是X-2
, stack的段地址是X-1
。 -
任務1-2
對程序task1_2.asm進行匯編、連接,用debug加載、跟蹤調試,基於結果,回答問題。
task1_2.asmassume ds:data, cs:code, ss:stack data segment db 4 dup(0) ; 預留4個字節單元,初始值均為0 data ends stack segment db 8 dup(0) ; 預留8個字節單元,初始值均為0 stack ends code segment start: mov ax, data mov ds, ax mov ax, stack mov ss, ax mov sp, 8 ; 設置棧頂 mov ah, 4ch int 21h code ends end start
① 在debug中將執行到line17結束、line19之前,記錄此時:寄存器(DS) =
076A
, 寄存器(SS) =076B
, 寄存器(CS) =076C
② 假設程序加載后,code段的段地址是X,則,data段的段地址是X-2
, stack的段地址是X-1
。 -
任務1-3
對程序task1_3.asm進行匯編、連接,用debug加載、跟蹤調試,基於結果,回答問題。
task1_3.asmassume ds:data, cs:code, ss:stack data segment db 20 dup(0) ; 預留20個字節單元,初始值均為0 data ends stack segment db 20 dup(0) ; 預留20個字節單元,初始值均為0 stack ends code segment start: mov ax, data mov ds, ax mov ax, stack mov ss, ax mov sp, 20 ; 設置初始棧頂 mov ah, 4ch int 21h code ends end start
① 在debug中將執行到line17結束、line19之前,記錄此時:寄存器(DS) =
076A
, 寄存器(SS) =076C
, 寄存器(CS) =076E
② 假設程序加載后,code段的段地址是X,則,data段的段地址是X-4
, stack的段地址是X-2
。 -
任務1-4
對程序task1_4.asm進行匯編、連接,用debug加載、跟蹤調試,基於結果,回答問題。
task1_4.asmassume ds:data, cs:code, ss:stack code segment start: mov ax, data mov ds, ax mov ax, stack mov ss, ax mov sp, 20 mov ah, 4ch int 21h code ends data segment db 20 dup(0) data ends stack segment db 20 dup(0) stack ends end start
① 在debug中將執行到line17結束、line19之前,記錄此時:寄存器(DS) =
076C
, 寄存器(SS) =076E
, 寄存器(CS) =076A
② 假設程序加載后,code段的段地址是X,則,data段的段地址是X+2
, stack的段地址是X+4
。 -
任務1-5
基於上述四個實驗任務的實踐、觀察,總結
① 對於如下定義的段,程序加載后,實際分配給該段的內存空間大小是16*[N/16]
。xxx segment db N dup(0) xxx ends
② 如果將程序task1_1.asm, task1_2.asm, task1_3.asm, task1_4.asm中,偽指令 end start 改成end , 哪一個程序仍然可以正確執行?結合實踐觀察得到的結論,分析、說明原因
task1_4.asm仍然可以正確執行, 前三個不行. end指令除了聲明程序結束,它還指明了程序的入口地址, task1_4.asm若不指明程序入口地址, 依舊是從start開始(程序段區域), 而前三個代碼的代碼首部並非程序段, 數據當做代碼運行, 不出意外應該是出錯的.
任務二
編寫一個匯編源程序,實現向內存單元b800:0f00 ~ b800:0f9f連續160字節,依次重復填充十六進制數據03 04。
assume cs:code
data segment
db 80 dup(03h, 04h)
data ends
code segment
start: mov ax, 0b800h
mov es,ax
mov ax, data
mov ds, ax
mov si, 0f00h
mov bx,0
mov cx,80
s: mov ax, [bx]
mov es:[bx+si],ax
add bx,2
loop s
mov ah, 4ch
int 21h
code ends
end start
任務三
已知8086匯編源程序task3.asm代碼片段如下。
要求:
① 編程實現把邏輯段data1和邏輯段data2的數據依次相加,結果保存到邏輯段data3中。
② 在debug中加載、反匯編、調試。在數據項依次相加前,和相加后,分別查看三個邏輯段data1,
data2, data3對應的內存空間,確認逐一相加后,結果的確保存在了邏輯段data3中。
task3.asm
assume cs:code
data1 segment
db 50, 48, 50, 50, 0, 48, 49, 0, 48, 49 ; ten numbers
data1 ends
data2 segment
db 0, 0, 0, 0, 47, 0, 0, 47, 0, 0 ; ten numbers
data2 ends
data3 segment
db 16 dup(0)
data3 ends
code segment
start:
mov bx, 0 ;偏移量
mov dx, 0 ;求和寄存器
mov cx, 10 ;10次循環
s:
mov dx, 0
mov ax, data1
mov ds, ax
add dl, [bx]
mov ax, data2
mov ds, ax
add dl, [bx]
mov ax, data3
mov ds, ax
mov [bx], dl
inc bx
loop s
mov ah, 4ch
int 21h
code ends
end start
相加前:
相加后:
data1
,data2
,data3
分別對應着076A, 076B, 076C 由圖中的結果可以看出 076A 和 076B 的數據加和並放到了 076C段中.
任務四
要求:
① 補全程序,實現把邏輯段data1中的8個字數據逆序存儲到邏輯段b中。
② 匯編、連接后,在debug中加載程序,運行到line15程序退出前,使用d命令查看數據段data2對應的內存空間。
assume cs:code
data1 segment
dw 2, 0, 4, 9, 2, 0, 1, 9
data1 ends
data2 segment
dw 8 dup(?)
data2 ends
code segment
start:
mov ax, data1
mov ds, ax
mov ax, data2
mov ss, ax
mov ax, 16
mov sp, ax
mov cx, 8
mov bx, 0
s:
push [bx]
add bx, 2
loop s
mov ah, 4ch
int 21h
code ends
end start
程序執行前:
程序執行后:
任務五
task5.asm
assume cs:code, ds:data
data segment
db 'Nuist'
; db 2, 3, 4, 5, 6
; db 5 dup(2)
db 5 dup(5)
data ends
code segment
start:
mov ax, data
mov ds, ax
mov ax, 0b800H
mov es, ax
mov cx, 5
mov si, 0
mov di, 0f00h
s: mov al, [si]
and al, 0dfh ; 11011111 變大寫
mov es:[di], al
mov al, [5+si]
mov es:[di+1], al
inc si
add di, 2
loop s
mov ah, 4ch
int 21h
code ends
end start
閱讀源程序,從理論上分析源代碼的功能,尤其是line15-25,循環實現的功能是什么,逐行理解每條指令的功能。
-
對程序進行匯編、鏈接,得到可執行文件,運行並觀察結果。
-
使用debug工具對程序進行調試,執行到程序返回前,即line25執行之后、line27執行之前,觀察結果。
-
源代碼中line19的作用是?
將字母變成大寫
and 11011111
-
修改line4里5個字節單元的值,重新匯編、鏈接、運行,觀察結果。
db 2, 3, 4, 5, 6 --> 改成: db 5 dup(2) 或 db 5 dup(5)
改成
db 5 dup(2)
后, 都變成第1個字符的樣式:00000010
改成
db 5 dup(5)
后, 都變成第4個字符的樣式:00000101
任務六
要求:
① 補全程序,將data段中的每行第一個單詞從大寫->小寫。
② 在debug中加載程序,反匯編,執行到line13退出前,用d命令查看data段對應的內存空間,確認每=行第一個單詞已經由大寫->小寫。
assume cs:code, ds:data
data segment
db 'Pink Floyd ' ; 16字節
db 'JOAN Baez ' ; 16字節
db 'NEIL Young ' ; 16字節
db 'Joan Lennon ' ; 16字節
data ends
code segment
start:
mov ax, data
mov ds, ax
mov cx, 4
mov bx, 0
upper_word:
mov dx, cx ; 存儲外層循環的cx
mov cx, 4
mov si, 0
upper_char:
or byte ptr [bx+si], 00100000b
inc si
loop upper_char
mov cx, dx
add bx, 16
loop upper_word
mov ah, 4ch
int 21h
code ends
end start
運行結果:
任務七
assume cs:codesg
data segment
db '1975', '1976', '1977', '1978', '1979'
dd 16, 22, 382, 1356, 2390
dw 3, 7, 9, 13, 28
data ends
table segment
db 5 dup( 16 dup(' ') )
table ends
codesg segment
start:
mov ax, data
mov ds, ax
mov ax, table
mov es, ax
;;;;;;;;;;;;;;;;;;;;放入年份;;;;;;;;;;;;;;;;;;;;
mov bx, 0 ; 第bx年
mov bp, 0 ; table中的位置
mov cx, 5
lay_years:
;內循環,放置一個年份
mov dx, cx ; 存儲外循環次數
mov cx, 4 ; 年份長度為4
mov si, 0 ;年份第si字符
lay_year:
mov al, ds:[bx+si]
mov es:[bp+si], al
inc si
loop lay_year
mov cx, dx
mov byte ptr es:[bp+4],' ' ;放一個空格
add bx, 4
add bp, 10h ; 一行16字節
loop lay_years
;;;;;;;;;;;;;;;;;;;;放入收入;;;;;;;;;;;;;;;;;;;;
mov bp, 0
mov cx, 5
lay_incomes:
; 字型數據(雙字節), 得移兩次
mov ax,ds:[bx] ; 由於連續存放在data中,bx指針繼續累加
mov es:[bp+5],ax
mov ax,ds:[bx+2]
mov es:[bp+7],ax
mov byte ptr es:[bp+9], ' ' ;放一個空格
add bx, 4
add bp, 10h
loop lay_incomes
;;;;;;;;;;;;;;;;;;;;放入雇員人數;;;;;;;;;;;;;;;;;;;;
mov bp, 0
mov cx, 5
lay_populations:
mov ax, ds:[bx] ;由於連續存放在data中,bx指針繼續累加
mov es:[bp+0ah], ax
mov byte ptr es:[bp+0ch], ' ' ;放一個空格
add bx, 2
add bp, 10h
loop lay_populations
;;;;;;;;;;;;;;;;;;;;求人均收入並放入;;;;;;;;;;;;;;;;;;;;
mov bp,0
mov cx,5
lay_average_incomes:
mov ax, es:[bp+5]
mov dx, es:[bp+7]
div word ptr es:[bp+0ah]
mov es:[bp+0dh], ax
mov byte ptr es:[bp+0fh], ' ' ;放一個空格
add bp, 10h
loop lay_average_incomes
mov ax, 4c00h
int 21h
codesg ends
end start
注意給高位dx也賦值