原文:https://www.pediy.com/kssd/pediy10/58757.html
;============================== ;filename:top2o5a.asm ;date:2008/1/21 ;============================== assume cs:code,ds:data,ss:stack stack segment dw 64 dup(0) stack ends data segment str db 'I Love Masm!',0 m db 'M',0 i db 6 dup('0') data ends code segment start: mov ax,data mov ds,ax mov ax,stack mov ss,ax mov sp,128 ;查找在字符串str中的首个与CH字符匹配的位置 lea si,str ;(SI)=str mov ch,ds:[m] ;(CH)=DS:[m] call string_char ;将位置值转换为一个字符串,方便输出 lea si,i ;(SI)=i call dtoc ;在第5行、第1列以白色输出父串 lea si,str mov dh,5 mov dl,1 mov cl,1010b call show_str ;在第6行、第1列以白色输出要匹配的字符 lea si,m mov dh,6 mov dl,1 mov cl,1010b call show_str ;输出结果 lea si,i mov dh,7 mov dl,1 mov cl,1010b call show_str mov ax,4c00h int 21h ;================================================== ;名称:string_char ;功能:在一个串中查找给定字符的第一个匹配的位置 ;参数:(ch)=字符 ; ds:si指向字符串的首地址 ;返回:(ax)=匹配的位置 ; (ax)=0表示未找到匹配位置 ;================================================== string_char proc near push si push di mov di,si ;(DI)=(SI) s2: push cx mov cl,0 ;如果已搜索到字符串末尾,求找到,退出循环 mov ch,ds:[di] jcxz outl pop cx cmp byte ptr ds:[di],ch ;判断当前字符是否为等于要匹配的字符 jz ok2 ;DS:[DI]=(CH),搜索到匹配的字符,退出循环 inc di ;(DI)=(DI)+1指向下一个字符 jmp short s2 outl: mov ax,0 ;(AX)=0未找到,直接返回 pop di pop si ret ok2: sub di,si ;首个匹配的位置值为(DI)=(DI)-(SI)+1 inc di mov ax,di pop di pop si ret string_char endp ;================================================================= ;名称:dtoc ;功能:将word型数据转变为表示十进制数的字符串,字符串以0为结尾符 ;参数:(ax)=word型数据 ; ds:si指向字符串的首地址 ;返回:无 ;================================================================= dtoc proc near push ax push bx push cx push si push di mov di,si mov bx,10 mov cx,0 s: push cx mov cx,ax jcxz enddtoc pop cx mov dx,0 div bx mov ds:[di],dl add byte ptr ds:[di],30h inc di inc cx jmp short s enddtoc: pop cx mov byte ptr ds:[di],0 ;在字符串末尾补0 dec di ;(DI)=(DI)-1指向字符串最后一个字符 mov ax,cx ;(cx)=字符串的长度 mov bl,2 div bl mov cl,al ;(CL)=(AL)=(CX)/2 jcxz r ;如果(CX)=0时,会执行循环一次,然后(CX)=(CX)-1=0-1=FFFFH,发生溢出错误 ;故添加代码(cx)=0时不执行循环跳转指令jcxz r s1: mov al,ds:[di] ;求得余数之后,要将字符串逆序,把余数摆正 mov bl,ds:[si] mov ds:[di],bl mov ds:[si],al inc si ;(SI)=(SI)+1 dec di ;(DI)=(DI)-1 loop s1 r: pop di pop si pop cx pop bx pop ax ret dtoc endp ;=========================================================== ;名称:show_str ;功能:在指定的位置,用指定的颜色,显示一个用0结束的字符串 ;参数:(dh)=行号(取值范围0~24),(dl)=列号(取值范围0~79), ; (cl)=颜色,ds:si指向字符串的首地址 ;返回:无 ;=========================================================== show_str proc near push dx push si push di push cx push ax mov ax,0b800h mov es,ax mov ax,160 mul dh mov dh,0 add ax,dx add ax,dx sub ax,2 mov di,ax mov ah,cl output: mov ch,ds:[si] mov cl,0 jcxz ok mov byte ptr es:[di],ch mov byte ptr es:[di+1],ah inc si inc di inc di jmp short output ok: pop ax pop cx pop di pop si pop dx ret show_str endp code ends end start
;============================== ;filename:top2o5b.asm ;date:2008/1/22 ;============================== assume cs:code,ds:data,ss:stack stack segment dw 64 dup(0) stack ends data segment str1 db 'I Love Win32 Assembly Language!',100 dup(0) str2 db 'I Love 80X86 Assembly Language!',0 data ends code segment start: mov ax,data mov ds,ax mov ax,stack mov ss,ax mov sp,128 ;将目的串str2连接到源串str1 lea di,str1 ;(DI)=str1 取目的串的偏移地址 lea si,str2 ;(SI)=str2 取源串的偏移地址 call string_cat ;在第10行、第10列输出连接后的字符串 lea si,str1 ;(SI)=str1 字符串首地址 mov dh,10 ;(DH)=10 第10行 mov dl,10 ;(DL)=10 第10列 mov cl,00001010b ;(CL)=10001010B 字符串属性为高亮、绿色 call show_str mov ax,4c00h int 21h ;==================================== ;名称:string_cat ;功能:将源字符串拼接到目的字符串 ;参数:ds:si指向源字符串的首地址 ; ds:di指向目的字符串的首地址 ;返回:无 ;==================================== string_cat proc near ;保护现场 push ax push si push di ;将DI指向目的数组的末尾0 s1: cmp byte ptr ds:[di],0 ;DS:[DI]==0时表示已到字符串末尾 jz s2 inc di ;(DI)=(DI)+1 DI指向下一个字符 jmp short s1 ;将源数组SI连接到目的数组,循环终止条件为SI指向数组末尾0 s2: cmp byte ptr ds:[si],0 ;DS:[SI]==0时表示已将源串的每一个字符连接到目的串 jz ok1 mov al,ds:[si] ;DS:[DI]=DS:[SI] mov ds:[di],al inc si ;(SI)=(SI)+1 SI指向下一个待连接的字符 inc di ;(DI)=(DI)+1 DI指向下一个单元 jmp short s2 ok1: mov byte ptr ds:[di],0 ;在目的串添加终止符0,因为不能保证每次连接字符串后目的串的末尾字符为0 ;恢复现场 pop di pop si pop ax ret string_cat endp ;=========================================================== ;名称:show_str ;功能:在指定的位置,用指定的颜色,显示一个用0结束的字符串 ;参数:(dh)=行号(取值范围0~24),(dl)=列号(取值范围0~79), ; (cl)=颜色,ds:si指向字符串的首地址 ;返回:无 ;=========================================================== show_str proc near push dx push si push di push cx push ax mov ax,0b800h mov es,ax mov ax,160 mul dh mov dh,0 add ax,dx add ax,dx sub ax,2 mov di,ax mov ah,cl output: mov ch,ds:[si] mov cl,0 jcxz ok mov byte ptr es:[di],ch mov byte ptr es:[di+1],ah inc si inc di inc di jmp short output ok: pop ax pop cx pop di pop si pop dx ret show_str endp code ends end start
;============================== ;filename:top2o5c.asm ;date:2008/1/22 ;============================== assume cs:code,ds:data,ss:stack stack segment dw 64 dup(0) stack ends data segment str1 db 'I Love 80X86 Assembly Language!',0 str2 db 'I Love Win32 Assembly Language!',0 buf db 6 dup(0) data ends code segment start: mov ax,data mov ds,ax mov ax,stack mov ss,ax mov sp,128 ;比较字符串str1、str2 lea si,str1 ;(SI)=str1 lea di,str2 ;(DI)=str2 call string_compare ;将结果转换为字符串并存储在buf中 lea si,buf ;(SI)=buf call dtoc ;在第5行、第1列以高亮、绿色输出结果 lea si,str1 ;(SI)=buf 要输出的字符串 mov dh,5 ;(DH)=5 第5行 mov dl,1 ;(DL)=1 第5列 mov cl,00001010b ;(CL)=0001010B 高亮、绿色 call show_str ;在第6行、第1列以高亮、绿色输出结果 lea si,str2 ;(SI)=buf 要输出的字符串 mov dh,6 ;(DH)=5 第5行 mov dl,1 ;(DL)=1 第5列 mov cl,00001010b ;(CL)=0001010B 高亮、绿色 call show_str ;在第7行、第1列以高亮、绿色输出结果 lea si,buf ;(SI)=buf 要输出的字符串 mov dh,7 ;(DH)=5 第5行 mov dl,1 ;(DL)=1 第5列 mov cl,00001010b ;(CL)=0001010B 高亮、绿色 call show_str mov ax,4c00h int 21h mov ax,4c00h int 21h ;================================================ ;名称:string_compare ;功能:比较两个字符串 ;参数:ds:si指向第一个字符串的首地址 ; ds:di指向第二个字符串的首地址 ;返回:(ax)=0 两个字符串相等 ; (ax)=1 第一个字符串大于第二个字符串 ; (ax)=-1 第一个字符串小于第二字符串 ;================================================ string_compare proc near ;保护现场 push bx push si push di mov ax,0 ;(AX)=0 假设两个字符串相等 ;循环流程图,逻辑比较简单,但应该还可以画得更美观一点 ; |yes→[DI]==0?|yes→第一个字符串等于第二个字符串 ; | |no →第一个字符串小于第二个字符串 ;[SI]==0?| ; | ; |no →[DI]==0?|yes→第一个字符串大于第二个字符串 ; |no →SI++ DI++ →判断两个字符的大小,或跳转循环,或返回[SI]==0?处继续执行 s4: cmp byte ptr ds:[si],0 ;[SI]==0? jz s2 ;[SI]==0 跳转到s2 jmp s3 ;[SI]!=0 跳转到s3 s2: cmp byte ptr ds:[di],0 ;[DI]==0? jz ok1 ;[DI]==0 跳转到ok1,表示两个字符串相等 jmp short less ;[DI]!=0 跳转到less,表示第一个字符串小于第二个字符串 s3: cmp byte ptr ds:[di],0 ;[DI]==0? jz greater ;[DI]==0 跳转到greater,表示第一个字符串大于第二个字符串 ;[DI]!=0 判断两个字符大小 mov bh,ds:[di] ;(BH)=DS:[DI] cmp byte ptr ds:[si],bh jg greater ;DS:[SI]>DS:[DI] 第一个字符串大于第二个字符串 jl less ;DS:[SI]<DS:[DI] 第一个字符串小于第二个字符串 inc si inc di jmp short s4 ;返回s4处继续分析比较 ;第一字符串大于第二字符串 greater: mov ax,1 ;(AX)=1 表示第一个字符串大于第二个字符串 pop di pop si pop bx ret ;第一个字符串小于第二字符串 less: mov ax,0ffffh ;(AX)=FFFFH=-1 表示第一个字符串小于第二个字符串 ok1: ;恢复现场 pop di pop si pop bx ret string_compare endp ;================================================================= ;名称:dtoc ;功能:将word型数据转变为表示十进制数的字符串,字符串以0为结尾符 ;参数:(ax)=word型数据 ; ds:si指向字符串的首地址 ;返回:无 ;================================================================= dtoc proc near push ax push bx push cx push si push di mov di,si mov bx,10 mov cx,0 s: push cx mov cx,ax jcxz enddtoc pop cx mov dx,0 div bx mov ds:[di],dl add byte ptr ds:[di],30h inc di inc cx jmp short s enddtoc: pop cx mov byte ptr ds:[di],0 ;在字符串末尾补0 dec di ;(DI)=(DI)-1指向字符串最后一个字符 mov ax,cx ;(cx)=字符串的长度 mov bl,2 div bl mov cl,al ;(CL)=(AL)=(CX)/2 s1: mov al,ds:[di] ;求得余数之后,要将字符串逆序,把余数摆正 mov bl,ds:[si] mov ds:[di],bl mov ds:[si],al inc si ;(SI)=(SI)+1 dec di ;(DI)=(DI)-1 loop s1 pop di pop si pop cx pop bx pop ax ret dtoc endp ;=========================================================== ;名称:show_str ;功能:在指定的位置,用指定的颜色,显示一个用0结束的字符串 ;参数:(dh)=行号(取值范围0~24),(dl)=列号(取值范围0~79), ; (cl)=颜色,ds:si指向字符串的首地址 ;返回:无 ;=========================================================== show_str proc near push dx push si push di push cx push ax mov ax,0b800h mov es,ax mov ax,160 mul dh mov dh,0 add ax,dx add ax,dx sub ax,2 mov di,ax mov ah,cl output: mov ch,ds:[si] mov cl,0 jcxz ok mov byte ptr es:[di],ch mov byte ptr es:[di+1],ah inc si inc di inc di jmp short output ok: pop ax pop cx pop di pop si pop dx ret show_str endp code ends end start