汇编:结构体

在32位汇编中,结构体(structures)用于组织和管理复杂的数据类型,结构体可以包含多个不同类型的数据项(成员);在MASM(Microsoft Macro Assembler)中,使用结构体可以让我们更有条理地处理数据。下面是一个定义和使用结构体的示例。

结构体定义格式
STRUCT_NAME STRUCT
    MEMBER_NAME1 TYPE1 [NUMBER_OF_ELEMENTS1] [INITIAL_VALUE1]
    MEMBER_NAME2 TYPE2 [NUMBER_OF_ELEMENTS2] [INITIAL_VALUE2]
    ; 更多成员...
STRUCT_NAME ENDS

STRUCT_NAME: 结构体的名称。

MEMBER_NAME: 结构体成员的名称。

TYPE: 成员的数据类型,如 BYTE, WORD, DWORD, DB, DW, DD 等。

NUMBER_OF_ELEMENTS: 可选,成员的元素个数,用于定义数组。

INITIAL_VALUE: 可选,成员的初始值。

示例:假设我们要定义一个表示人的结构体,其中包含姓名(字符串)和年龄(整数)。
Person struct 
    szName db 32 dup(?)
    nAge dd ?
Person ends

在这个示例中,Person结构体包含两个成员:

  • name: 一个32字节的字符数组,用于存储人的姓名。

  • age: 一个32位的整数,用于存储人的年龄。

声明和初始化结构体变量
.data
wolven Person <"wolven Chan",24>

这行代码声明了一个名为wolvenPerson结构体变量,并初始化其szName"wolven Chan"nAge24

访问结构体成员
.code
main proc
    mov eax,wolven.nAge
    mov ebx,offset wolven.szName
main endp

wolvennAge成员的值移动到eax寄存器;将wolvenszName成员的地址偏移值移动到ebx寄存器。

程序执行完后的EAXEBX寄存器分别为上述结构体变量初始化时的值。

示例代码:

代码展示了如何定义一个结构体,并通过自定义过程printfS输出结构体的成员。

.586
.model flat,stdcall
option casemap:none
​
;结构体定义
Person struct 
    szName db 32 dup(?)
    nAge dd ?
Person ends
​
;数据段
.data
szFormatS db '%s',0
wolven Person <"wolven Chan",24>
​
;代码段
.code
;①自定义过程,打印字符串
printfS proc szFormat:DWORD,szValue:DWORD   
    xor eax,eax
    mov eax,szValue
    push eax
    mov eax,szFormat
    push szFormat
    call printf
    add esp,8
    ret
printfS endp
​
;②程序入口
main proc
    mov eax,wolven.nAge
    mov ebx,offset wolven.szName
    invoke printfS,offset szFormatS,ebx
main endp
end 
结构体定义:

Person struct: 定义了一个名为Person的结构体。

szName db 32 dup(?): 一个32字节的字符数组,用于存储姓名。

nAge dd ?: 一个32位的整数,用于存储年龄。

Person ends: 结束结构体定义。

数据段:

szFormatS db '%s',0: 定义了一个以零结尾的字符串,表示格式化字符串%s,用于打印字符串。

wolven Person <"wolven Chan",24>: 定义了一个名为wolvenPerson结构体变量,并初始化szName"wolven Chan"nAge24

代码段
①自定义过程

printfS proc szFormat:DWORD,szValue:DWORD: 定义一个名为printfS的过程,接受两个参数,szFormatszValue

xor eax,eax: 将eax寄存器清零。

mov eax,szValue: 将第二个参数szValue的值移动到eax寄存器。

push eax: 将eax的值压入堆栈。

mov eax,szFormat: 将第一个参数szFormat的值移动到eax寄存器。

push szFormat: 将szFormat的值压入堆栈。

call printf: 调用C运行时库的printf函数。

add esp,8: 调整堆栈指针,移除之前压入的两个参数(每个参数占4个字节)。

ret: 返回到调用printfS的地方。

为什么自定义过程要先将第二个参数压入栈中
复制代码
在汇编中,自定义过程(函数)使用的参数传递方式与调用约定有关;展示代码采用了stdcall调用约定,在这种约定中,参数是按从右到左的顺序压入堆栈的,这样函数就可以按照从左到右的顺序来访问参数。
②程序入口(main)

mov eax,wolven.nAge: 将结构体变量wolvennAge成员的值移动到eax寄存器。

mov ebx,offset wolven.szName: 将结构体变量wolvenszName成员的地址移动到ebx寄存器。

invoke printfS,offset szFormatS,ebx: 调用自定义过程printfS,传入szFormatSwolven.szName的地址。

最后结构体变量wolven的nAge成员值被放入eax中,且黑窗口中打印szName成员内容:

相关推荐
百年孤独_4 天前
对于基础汇编的趣味认识
汇编·性能优化
xiaozhiwise4 天前
ARM base instruction -- ccmp
汇编
向你扔鸡爪7 天前
Visual Studio-X64汇编编写
汇编·windows·visual studio
Lordaeron_ESZ7 天前
CSAPP Attack Lab
linux·汇编
xiaozhiwise8 天前
ARM base instruction -- sxtw
汇编
天赐细莲11 天前
C++的哲学思想
java·c语言·汇编·c++·python
王鑫的博客88611 天前
ARM基础知识点及简单汇编语法
linux·汇编·arm开发
洛书千年11 天前
linux驱动开发-arm汇编基础
汇编·arm开发·驱动开发
小咖拉眯11 天前
vscode将c++项目打包exe进行反汇编练习
c语言·汇编·c++·ide·vscode·安全·密码学
妖怪喜欢风14 天前
Linux 5.0在start_kernel之前做了什么事?(以aarch64为例)
linux·汇编