学 Win32 汇编[12]: PTR、OFFSET、ADDR、THIS

PTR: 指定要操作的数据尺寸
; Test12_1.asm
.386
.model flat, stdcall

include    windows.inc
include    kernel32.inc
include    masm32.inc
include    debug.inc
includelib kernel32.lib
includelib masm32.lib
includelib debug.lib

.data
    val db 11h, 22h, 33h, 44h, 55h, 66h, 77h, 88h

.code
main proc
    xor eax, eax             ;清空 EAX, 同 mov eax, 0
    mov eax, dword ptr val   ;
    PrintHex eax             ;44332211
    
    xor eax, eax             ;
    mov eax, dword ptr val+1 ;
    PrintHex eax             ;55443322
    
    xor eax, eax             ;
    mov ax, word ptr val     ;
    PrintHex eax             ;00002211
    
    xor eax, eax             ;
    mov al, byte ptr val     ;
    PrintHex eax             ;00000011
    ret
main endp
end main

  
  

OFFSET: 获取全局变量或标号的偏移地址

; Test12_2.asm
.386
.model flat, stdcall

include    windows.inc
include    kernel32.inc
include    masm32.inc
include    debug.inc
includelib kernel32.lib
includelib masm32.lib
includelib debug.lib

.data
    v1 db 'abcdefg', 0
    v2 dd 11223344h

.code
main proc
    PrintHex offset v1    ;00403000
    PrintHex offset v2    ;00403008
    PrintHex offset main  ;00401000 - 这里的 main 是个标号
    ret
;本例中的 offset 不能用 addr 代替
main endp
end main

  
  

ADDR: 类似 offset 也是获取变量的地址...

; Test12_3.asm
.386
.model flat, stdcall

;include    windows.inc
include    kernel32.inc
includelib kernel32.lib
include    user32.inc
includelib user32.lib

.data
    v1 dd 00434241h ;ABC
    v2 dd 00636261h ;abc

.code
main proc
    invoke MessageBox, 0, offset v1, offset v2, 0 ;现在 v1、v2 是全局变量
    invoke MessageBox, 0,   addr v2,   addr v1, 0 ;使用 offset 和 addr 均可
    invoke ExitProcess, 0
main endp
end main

  
  

获取局部变量的地址只能使用 ADDR:

; Test12_4.asm
.386
.model flat, stdcall

;include    windows.inc
include    kernel32.inc
includelib kernel32.lib
include    user32.inc
includelib user32.lib

.code
main proc
    LOCAL v1,v2
    mov v1, 00434241h
    mov v2, 00636261h
    ;invoke MessageBox, 0, offset v1, offset v2, 0  ;offset 不能获取局部变量的地址
    invoke MessageBox, 0,   addr v2,   addr v1, 0
    invoke ExitProcess, 0
main endp
end main

  
  

OFFSET 和 ADDR 的异同:
1、offset 不能获取局部变量的地址;
2、addr 只能用于调用函数(invoke)时, 不能用于赋值操作;
3、addr 面对局部变量时会转换为 lea 等指令, addr 面对全局变量时则直接调用 offset;
4、在 invoke 中应尽量使用 addr, 其他只用 offset.


THIS:

; Test12_5.asm
.386
.model flat, stdcall

include    windows.inc
include    kernel32.inc
include    masm32.inc
include    debug.inc
includelib kernel32.lib
includelib masm32.lib
includelib debug.lib

.data
    TextAddr equ this byte   ;伪指令 this 可让当前变量和下一个变量同址 
    szText db 'Asm', 0
.code
main proc
    PrintHex offset szText   ;00403000
    PrintHex offset TextAddr ;00403000
    
    PrintString szText       ;Asm
    mov [TextAddr], 'a'      ;给 TextAddr 赋值
    PrintString szText       ;asm
    ret
main endp
end main

  
  
原文:http://www.cnblogs.com/del/archive/2010/04/07/1706340.html
以下是一个用MASM编写的汇编程序,可以统计指定数据区内的英文字母、英文单词和行数: ``` .model flat, stdcall .stack 4096 .data buffer db 10000 dup(0) letters db 26 dup(0) wordCount dd 0 lineCount dd 1 inWord db 0 .code main proc ; 读取文件到缓冲区 call readFile ; 统计字母和单词 call countLettersAndWords ; 输出结果 call printResult ; 退出程序 invoke ExitProcess, 0 main endp ; 读取文件到缓冲区 readFile proc ; 打开文件 invoke CreateFile, addr fileName, GENERIC_READ, 0, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0 ; 获取文件大小 invoke GetFileSize, eax, 0 ; 创建文件映射 invoke CreateFileMapping, eax, 0, PAGE_READONLY, 0, 0, 0 ; 映射文件到内存 invoke MapViewOfFile, eax, FILE_MAP_READ, 0, 0, 0 ; 将文件内容复制到缓冲区 mov esi, eax mov edi, offset buffer mov ecx, [filesize] rep movsb ; 关闭文件映射 invoke CloseHandle, eax ; 关闭文件句柄 invoke CloseHandle, eax ret readFile endp ; 统计字母和单词 countLettersAndWords proc mov ecx, offset buffer mov edx, offset letters mov ebx, 0 ; ebx用于计数 mov [inWord], 0 ; 标记是否在单词中 ; 循环处理每个字符 loop1: mov al, [ecx] ; 如果是字母,增加对应字母的计数器 cmp al, 'A' jl notLetter cmp al, 'Z' jg notLetter add byte ptr [edx+al-'A'], 1 ; 如果不在单词中,将标记设置为在单词中,并增加单词计数器 cmp [inWord], 0 jne inWord mov [inWord], 1 inc [wordCount] inWord: jmp nextChar notLetter: ; 如果不是字母,将标记设置为不在单词中 mov [inWord], 0 ; 如果是空格或换行符,增加行数计数器 cmp al, ' ' jne notSpace cmp [inWord], 1 jne notWordEnd spaceOrNewLine: mov [inWord], 0 inc [lineCount] notWordEnd: jmp nextChar notSpace: cmp al, 0x0A jne nextChar jmp spaceOrNewLine nextChar: inc ecx cmp byte ptr [ecx], 0 jne loop1 ret countLettersAndWords endp ; 输出结果 printResult proc ; 输出字母计数器的值 mov edx, offset letters mov ecx, 26 mov esi, offset letterNames printLetters: mov al, [edx] mov bl, [esi] invoke printf, addr letterFormat, bl, al inc edx add esi, 2 loop printLetters ; 输出单词计数器的值 invoke printf, addr wordFormat, [wordCount] ; 输出行数计数器的值 invoke printf, addr lineFormat, [lineCount] ret printResult endp .data fileName db "input.txt", 0 filesize dd ? letterNames db "A ", "B ", "C ", "D ", "E ", "F ", "G ", "H ", "I ", "J ", "K ", "L ", "M ", "N ", "O ", "P ", "Q ", "R ", "S ", "T ", "U ", "V ", "W ", "X ", "Y ", "Z ", 0 letterFormat db "%s%d\n", 0 wordFormat db "Word count: %d\n", 0 lineFormat db "Line count: %d\n", 0 end main ``` 请注意,这只是一个简单的示例程序,可能需要根据实际需求进行修改。此外,为了使程序能够正确运行,您需要在编译时链接 C 标准库,以便使用 printf 函数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值