汇编 字符串处理

字符串处理

前言

按下列要求编程:1)从键盘输入一个字符串(串长不大于80);2)以十进制输出字符串中非字母字符的个数(不是az或AZ);3)输出原字符串且令非字母字符闪烁显示;4)找出字符串中ASCII码值最大的字符,在字符串中用红色显示;5)数据的输入和结果的输出都要有必要的提示,而且提示独占一行;6)要使用到子程序。


提示:以下是本篇文章正文内容,下面案例仅供参考

代码如下 :

data segment
   hintinput db "Input string:$"
   hintoutput1 db "Output the number of non-alpha characters:$";
   hintoutput2 db "Non-alphabetical characters flashing:$";
   hintoutput3 db "maxth ch:$";
           buf db 200
	           db ?
	           db 200 dup(?)
         count db 0
crlf db 0ah,0dh,'$'
data ends
stack segment stack
   string db 500 dup(?)
stack ends
extra segment   ;定义附加段

extra ends
codes segment
	assume cs:codes,ds:data,ss:stack,es:extra
	main proc far
start:
mov ax,data  ;分别将数据段,堆栈段,附加段送入代码段
  mov ds,ax
  mov ax,stack
  mov ss,ax
  mov ax,extra
  mov es,ax

  lea dx,hintinput ;输入提示语
  mov ah,9;输出功能入口在dx
  int 21h
  lea dx,crlf;输入提示语后,回车换行
  mov ah,09
  int 21h

  mov ah,0ah;输入功能入口在dx
  lea dx,buf
  int 21h

  lea dx,crlf        
  mov ah,9
  int 21h
;------------------------------------首先用cl储存字符串长度,si源变址寄存器指向串真正开始的地方
  mov cl,buf+1;cl中放置实际字符串长度
                               
  lea si,buf+2;si放置字符串首地址
;------------------------------------首先输出提示语回车换行,以十进制输出字符串中非字母字符的个数(不是a to z或 A to Z)。
  
  call print1

  call print2

  call print3

  mov ah,4ch
  int 21h
  main endp
  
print1 proc near                        
  push cx
  push si
  lea dx,hintoutput1 ;输出提示语
  mov ah,09h;输出功能入口在dx
  int 21h
  lea dx,crlf;输入提示语后,回车换行
  mov ah,09
  int 21h
  mov dl,0 ;用dl来计数,统计符串中非字母字符的个数
  L0:
  mov al,[si]
  cmp al,'A'      
  jb L1           ;
  cmp al,'Z'      
  jbe L2           ;
  cmp al,'a'      
  jb L1           ;
  cmp al,'z'      
  jbe L2           ;

L1:
  inc dl     ;计数器加1
L2:
  inc si
  dec cl
  jnz L0
;----------------------------------输出非字母字符的个数dl

  cmp dl,9h
  jbe L             ;如果非字母个数小于10,则直接加30输出
  mov dh,0           ;否则把dx存入ax中
  mov ax,dx
  mov bl,10               ;ax除以10后,ah作为商
  div bl                              ;al作为余数
  mov dl,al
  mov ch,al           ;防止ah中的商被02h冲掉,先把商转移到ch中
  add dl,30h            ;先输出商,再输出余数,就是10进制了
  mov ah,02h
  int 21h

  mov dl,ch
L:
  add dl,30h
  mov ah,02h
  int 21h
  lea dx,crlf     ;输出后,回车换行
  mov ah,09
  int 21h
  pop si
  pop cx
  ret
  
print2 proc near
  lea dx,hintoutput2 ;输出提示语
  mov ah,09h         ;输出功能入口在dx
  int 21h
  lea dx,crlf        ;输入提示语后,回车换行
  mov ah,09h
  int 21h
  push cx
  push si
  cld                ;方向标志位df清零
  
L3: 
  push cx 
  lodsb            ;从字符串串中取数据至al
  cmp al,'A'      
  jb L4            ;如果字符<'A',跳转到L4
  cmp al,'Z'      
  jbe L5           ;如果字符<='Z',跳转到L5
  cmp al,'a'      
  jb L4            ;如果字符<'a',跳转到L4
  cmp al,'z'      
  jbe L5           ;如果字符<='Z',跳转到L5
L4:                ;如果不是字母,则进行闪烁输出
   mov bl,10000111b  ;bl属性闪烁输出
   mov bh,0        ;显示页为0
   mov cx,1        ;显示字符为1个
   mov ah,09h
   int 10h         ;输出bl属性的字符串
   mov ah,03h
   int 10h         ;读光标位置
   inc dl          ;输出列+1
   mov ah,02h
   int 10h         ;置光标位置
   jmp L6
L5:
   mov bl,00000111b   ;bl属性为平常输出
   mov bh,0            ;显示页为0
   mov cx,1           ;显示字符个数为1个
   mov ah,09h
   int 10h             ;输出属性为bl的字符
   mov ah,03h
   int 10h             ;读光标位置
   inc dl              ;输出列+1
   mov ah,02h
   int 10h             ;置光标位置
L6:
  pop cx
  loop L3
  
  lea dx,crlf           ;输入提示语后,回车换行
  mov ah,09
  int 21h
  pop si
  pop cx
  ret
print3 proc near
  lea dx,hintoutput3  ;输出提示语
  mov ah,09h          ;输出功能入口在dx
  int 21h
  lea dx,crlf         ;输入提示语后,回车换行
  mov ah,09
  int 21h
  push cx
  push si
            ;count用来存放ascill最大的字符
L7:   
  mov al,[si]
  cmp al,count       ;比较al和count的大小,如果al>count,则令al=count
  jbe L8
  mov count,al        
L8:  
  inc si
  dec cl
  jnz L7
  pop si
  pop cx
  cld
 L9: 
  push cx 
  lodsb               ;从串中取数据至al
  cmp al,count      
  jz L11           ;如果字符==ascill码最大字符
L10:                
   mov bl,00000111b   ;bl属性正常输出
   mov bh,0           ;显示页为0
   mov cx,1           ;显示字符为1个
   mov ah,09h
   int 10h            ;输出bl属性的字符串
   mov ah,03h
   int 10h             ;读光标位置
   inc dl             ;输出列+1
   mov ah,02h
   int 10h            ;置光标位置
   jmp L12
L11:
   mov bl,00000100b   ;bl属性为红色输出
   mov bh,0           ;显示页为0
   mov cx,1           ;显示字符个数为1个
   mov ah,09h
   int 10h            ;输出属性为bl的字符
   mov ah,03h
   int 10h            ;读光标位置
   inc dl              ;输出列+1
   mov ah,02h
   int 10h            ;置光标位置
L12:
  pop cx
  loop L9
  lea dx,crlf        ;输出后,回车换行
  mov ah,09
  int 21h
  
  ret
codes ends
end start
;
  • 2
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
汇编语言中,字符串输入通常需要使用系统调用来实现。不同的操作系统提供的系统调用可能不同,下面以Linux操作系统为例,介绍一种实现字符串输入的方法。 在Linux系统中,字符串输入通常使用系统调用read实现。read系统调用的参数包括文件描述符,读取的缓冲区地址和大小。对于标准输入设备(例如键盘),文件描述符为0。因此,我们可以通过以下汇编代码实现从键盘读取字符串: ``` section .data buffer db 100 ; 缓冲区大小为100字节 section .bss len resb 1 ; 存储读取的字符串长度 section .text global _start _start: ; 从标准输入设备读取字符串 mov eax, 3 ; 系统调用号为3,即read mov ebx, 0 ; 文件描述符为0,即标准输入设备 mov ecx, buffer ; 缓冲区地址 mov edx, 100 ; 缓冲区大小 int 0x80 ; 执行系统调用 ; 计算读取的字符串长度 mov esi, buffer ; 将缓冲区地址存入esi寄存器 xor ecx, ecx ; 将ecx寄存器清零 loop: cmp byte [esi+ecx], 0 ; 比较esi寄存器指向的地址处是否为0 je done ; 如果是,则跳转到done标签 inc ecx ; 将ecx寄存器加1 jmp loop ; 跳转到loop标签 done: mov [len], cl ; 将ecx寄存器的值存储到len变量中 ; 其他操作 ; ... ``` 上面的代码定义了一个100字节大小的缓冲区buffer和一个用于存储字符串长度的变量len。在_start标签处,我们使用read系统调用从标准输入设备读取字符串,并将其存储在buffer中。然后,我们使用一个循环计算读取的字符串长度,将其存储在len变量中。在计算字符串长度时,我们使用了一个类似于C语言中的指针操作——将缓冲区地址存入esi寄存器,然后使用[esi+ecx]的方式访问缓冲区中的每个字符。 需要注意的是,上面的代码仅为示例,实际应用中可能需要添加一些异常处理逻辑,例如输入不合法或输入超出缓冲区大小等情况。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值