1、北京理工大学汇编语言实验报告实验三字符串操作实验北京理工大学汇编语言实验报告实验三字符串操作实验实验三 字符串操作实验一、实验目的1)熟悉串操作指令的功能与应用;2)掌握串操作指令的寻址方式及使用方法,编写常用的字符串处理程序;3)了解汇编语言字符串处理基本流程;二、实验软硬件环境1)硬件环境:惠普64位一体化计算机及局域网;2)软件环境:windows8,红蜘蛛管理系统,MASMforWindows。三、实验相关知识1)字符串操作流程 SI寄存器保存源串首地址; DI寄存器保存目的串首地址; CX寄存器保存字符串长度; CLD或STD指令设置字符串处理方向; 当CLD指令使DF=0,在执行串
2、处理指令时可使地址自动增量;STD使DF=1,在执行串处理指令时可使地址自动减量。2)重复前缀指令 重复次数由计数寄存器CX中存放的值决定,指令每重复执行一次,计数器CX中值减1,当CX中值减至0时,停止重复执行,继续执行下一条指令。 当REP无条件重复前缀,重复串操作直到计数寄存器的内容CX为0为止。经常与REP配合工作的字符串处理指令有MOVS、STOS和LODS。 当REPE/REPZ判断计数寄存器的内容CX是否为0或ZF=0(即比较的两个操作数不等),只要满足一个则重复执行结束,否则继续执行。可以与REPE/REPZ配合工作的串指令有CMPS和SCAS。 当REPNE/REPNZ判断计
3、数寄存器的内容是否为0或ZF=1(即比较的两个操作数相等),只要满足一个则重复执行结束,否则继续执行。可以与REPE/REPZ配合工作的串指令有CMPS和SCAS。3)字符串操作指令 lodsb、lodsw:把DS:SI指向的存储单元中的数据装入AL或AX,然后根据DF标志增减SI; stosb、stosw:把AL或AX中的数据装入ES:DI指向的存储单元,然后根据DF标志增减DI; movsb、movsw:把DS:SI指向的存储单元中的数据装入ES:DI指向的存储单元中,然后根据DF标志分别增减SI和DI; scasb、scasw:把AL或AX中的数据与ES:DI指向的存储单元中的数据相减,
4、影响标志位,然后根据DF标志分别增减SI和DI; cmpsb、cmpsw:把DS:SI指向的存储单元中的数据与ES:DI指向的存储单元中的数据相减,影响标志位,然后根据DF标志分别增减SI和DI; rep:重复其后的串操作指令。重复前先判断CX是否为0,为0就结束重复,否则CX减1,重复其后的串操作指令。主要用在MOVS和STOS前。一般不用在LODS前。上述指令涉及的寄存器:段寄存器DS和ES、变址寄存器SI和DI、累加器AX、计数器 CX涉及的标志位:DF、AF、CF、OF、PF、SF、ZF。4、实验内容1)编写程序,比较两个字符串BUF1和BUF所含的字符是否相同,相同则AL返回0,不同
5、AL返回1,字符串长度要求自动获取,要求用字符串处理方法。提示:输入两个字符串之后,将串操作所必须的寄存器等参数设置好,然后使用串操作指令进行从头到尾的比较,两个字符串相等的条件是串长度相等且对应的字符相同。(I)实验框图(II)实验代码DATAS SEGMENT BUF1 DB ABCDEFGH COUNT1 EQU $-BUF1 ;利用EQU指令,自动获取字符串长度 DATAS ENDSEXTRA SEGMENT ;定义附加段,即ES段 BUF2 DB ABCDEFGH COUNT2 EQU $-BUF2 ;功能同BUF1EXTRA ENDS CODES SEGMENT ASSUME CS
6、:CODES,DS:DATAS,ES:EXTRASTART: MOV AX,DATAS MOV DS,AX MOV AX,EXTRA MOV ES,AX LEA SI,BUF1 ;把源操作串的地址放在SI中 LEA DI,BUF2 ;把目的操作串的地址放在DI中 MOV CX,COUNT1 CMP CX,COUNT2 ;先比较BUF1、BUF2的长度 JNZ EXIT1 ;长度不一样,字符串不同,跳转到EXIT1 CLD ;设置字符串操作方向 REPE CMPSB ;逐个字符比较 JNZ EXIT1 ;一旦有不同的字符,跳转到EXIT1 MOV AL,0 ;若全部相同,则字符串相同,返回AL=
7、0 JMP EXIT2 EXIT1: MOV AL,1 ;当字符串不同时,返回AL=1 EXIT2: MOV AH,4CH INT 21HCODES ENDSEND START(III)实验结果序号BUF1BUF2AL1ABCDEFGHABCDEFGH02ABCABCDE13ABCABD1(IV)实验结果分析序号1中,程序先经过比较字符串长度,相等后再逐一判断每个字符,确定都相等后,返回AL=0;序号2中,比较字符串长度已经发现不相等,所以返回AL=1;序号3中,比较了字符串长度,发现相等,进一步比较字符,循环到最后一个字符时,不相等,所以AL=1.2)编写程序,设有一字符串存放在以BUF为首
8、址的数据区中,其最后一字符$作为结束标志,计算该字符串的长度并输出。提示:从串的第一个字符开始统计,直到遇到定义的字符串结束符为止,看看在这个过程中总共有多少个字符,即求得串的长度。(I)实验框图(II)实验代码DATAS SEGMENT BUF DB TBE123000000 $ DATAS ENDSEXTRA SEGMENT CHAR DB $ ;ES段设置待比较的字符$EXTRA ENDSCODES SEGMENT ASSUME CS:CODES,DS:DATAS,ES:EXTRASTART: MOV AX,DATAS MOV DS,AX MOV AX,EXTRA MOV ES,AX L
9、EA SI,BUF ;获取BUF的地址 L:MOV AL,SI SCASB CHAR JNZ EXIT1 JZ EXIT2 EXIT1: INC SI ;不是$,计数器加1,同时字符串后移一位 JMP L EXIT2: DEC SI ;SI多加了一个1 MOV AX, SI MOV BL,10 DIV BL ;考虑SI是两位数的情况 MOV DX,AX ADD DX,3030H ;把数字转化为ASCII码 MOV AH,02H INT 21H ;打印低位 MOV DL,DH MOV AH,02H INT 21H ;打印高位 MOV AH,4CH INT 21HCODES ENDS END ST
10、ART(III)实验结果序号字符串屏幕输出结果1123456$ 0620123456789$ 10(IV)实验结果分析序号1:L循环了6次,查找到$,故SI等于6,直接用DOS的2号功能打印出字符2,注意 DL里的数字2转化为“字符2的ASCII码”,加上30H即可;序号2:循环十次,主要是逻辑上的两位数,要分离处理,因为2号功能只能打印一个字符;考虑除以10,得到商作为十位,余数作为个位。此时还是二进制,加上3030H,在打印即可。3)编写程序,将内存中BUF1某一区域的数据传送到另一区域BUF2 中,要求用字符串处方法。(I)实验框图(II)实验代码DATAS SEGMENT BUF1 D
11、B TBE123000000 COUNT EQU ($-BUF1)DATAS ENDSEXTRA SEGMENT BUF2 DB COUNT DUP(?) ;定义了BUF1长度COUNT的字符串EXTRA ENDSCODES SEGMENT ASSUME CS:CODES,DS:DATAS,ES:EXTRASTART: MOV AX,DATAS MOV DS,AX MOV AX,EXTRA MOV ES,AX LEA SI,BUF1 LEA DI,BUF2 MOV CX,COUNT ;设置CX的值 CLD ;设置DF=0,SI增加的方向进行串操作 REP MOVSB ;进行数据段DS到附加段E
12、S的搬移 MOV AH,4CH INT 21HCODES ENDS END START(III)实验结果字符串BUF1结果TBE123000000 见下图(IV)实验分析本题难度较小,只要一个重复前缀REP,再用MOVSB就可以实现“搬移”。4)编写程序,在已知字符串中搜索特定字符#,若找到则AL返回0,找不到AL返回1,要求用字符串处理方法。(I)实验框图(II)实验代码DATAS SEGMENT BUF1 DB TBE123000000 COUNT EQU ($-BUF1)DATAS ENDSEXTRA SEGMENT CHAR DB #EXTRA ENDSCODES SEGMENT AS
13、SUME CS:CODES,DS:DATAS,ES:DATASSTART: MOV AX,DATAS MOV DS,AX MOV ES,AX LEA DI,BUF1 MOV AL,# ;#放在AL中,准备使用SCASB指令 MOV CX,COUNT ;设置循环次数 CLD ;设置DF=0,正向操作 REPNE SCASB ;不相等时继续循环,相等时顺序执行下面指令 JZ FOUND ;若ZF=0,说明找到#,赋值AL=0 MOV AL,1 ;否则,没找到#,赋值AL=1 JMP EXIT FOUND: MOV AL,0 EXIT: MOV AH,4CH INT 21HCODES ENDS EN
14、D START(III)实验结果序号字符串AL1TBE123000000 12TBE12300# 0BUF为TBE123000000的结果:BUF为TBE12300#的结果 :(IV)实验分析本次实验较为简单,设置好CX和DF后,将#放入AL,利用重复前缀REPNE以及字符串比较指令SCASB,进行逐个比较,一旦发现ZF=0,跳转到FOUND,对AL进行赋0;否则,比较结束后,没找到#,AL=1。5)编写程序,统计一串字符串中字符&出现的次数,要求用字符串处理方法。(I)实验框图(II)实验代码DATAS SEGMENT BUF1db &adc& count equ $-stringDATAS
15、 ENDSSTACKS SEGMENT STACKS ENDSCODES SEGMENT ASSUME CS:CODES,DS:DATAS,SS:STACKSSTART: MOV AX,DATAS MOV DS,AX mov bx,0 ;将计数器BX清零 lea si,BUF1 mov cx,count ;设置循环次数 Cld ;DF=0,正向操作 L1: lodsb cmp al,& jnz L2 ;不是&就继续循环 inc bx ;是&,BX就加1 L2 : loop L1 MOV AH,4CH INT 21HCODES ENDSEND START(III)实验结果序号字符串&个数(BX)
16、1&adc&42159*%¥#0(IV)实验分析将CX和DF设置好以后,就可以从第一个字符开始比较,这里用到CMP,而不是SCASB,因为前者可以用于比较并计数,后者只能扫描字符。找到一个&,就让BX加1,否则跳到LOOP L1,继续循环。6)编写程序,设有一字符串已存放在STRBUF为首址的数据区中,编一程序找出其中的#字符,如果字符串中不存在#字符则应该输出“Notfound”。(I)实验框图(II)实验代码DATAS SEGMENT strbuf db &AB#& count equ $-strbuf string1 db Not found$ string2 db Found$ ;可能
17、会输出的字符串DATAS ENDSCODES SEGMENT ASSUME CS:CODES,DS:DATASSTART: MOV AX,DATAS MOV DS,AX mov bx,0 lea si,strbuf ;获取字符串地址 mov cx,count ;设置循环次数 Cld ;设置DF=0 L1: lodsb cmp al,# jz L2 loop L1 mov ah,9h ;未找到#,打印Not found lea dx,string1 int 21h L2 : mov ah,9h lea dx,string2 int 21h ;找到#,打印Found MOV AH,4CH INT
18、21HCODES ENDSEND START(III)实验结果序号字符串结果1&ABCD&Not found2&AB#&Found(IV)实验分析本题与与第(4)题思路一样,只是最后结果的处理不一样,题目要求没找到#,输出Not found,需要调用DOS的9号功能来打印字符串;在此基础上,没找到时,输出Found,更加直观。5、实验心得 本次实验主要是要求熟练掌握字符串操作,在此基础上,还有一些打印输出的训练,比如打印单个字符、字符串。通过实验,我对字符串操作有了更清晰的思路;在此将字符串操作的主要流程归纳如下:(1)数据段中的源串首地址放入SI;(2)附加段中的目的串首地址放入DI;(3)数据串长度放入CX;(4)建立方向标志;(CLD 指令使DF=0,STD指令使DF=1)(5)执行串操作。 在第五步中,会用到重复前缀,REPREPZREPNZ等等,深入地了解了它们的循环条件后,会给编程带来很大便利。
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1