C语言好难——结构体的内存分配
来源:互联网 发布:雅思6 知乎 编辑:程序博客网 时间:2024/06/07 12:31
结构体的内存分配,我个人是觉得比较蛋疼的,它有一个需要遵循的原则,地址对齐,也有人称为内存对齐,叫法没关系,反正我只是“拿来”,会用就行。
好了,先有这么一个概念,什么是内存对齐,先丢一边。……~(~o ̄▽ ̄)~o 。。。滚来滚去……o~(_△_o~) ~。。
设char占用1个字节,int占用4个字节。
那么问题1
struct A
{
char a;
int a0;
}ma;
char a1;
a1 = sizeof(ma);
下面是在keil中仿真的结果
?_?,为什么是8???,不是5?
是的,不是5,在编译器里面,为了非连续变量取得的地址能整除于结构体的基地址(结构体的分配的基地址也需要满足可以整除于所占用的内存的最大数据类型的字节数,这句话好绕啊_(:3」∠)_),偏移地址需要对齐。就是说在给变量分配的时候,如果不是连续定义的变量,直接分配一个最大的内存空间,即使我给a0变量定义的是一个char型变量,编译器依旧给其分配了4个字节。
关于偏移地址对齐的说法,我个人倾向于同意的说法是,在给结构体分配内存空间时,如果变量地址不连续(注意这句话,很重要),编译器宁可取结构体中内存占用最大的元素为每个变量分配内存,不够的空间予以补足,只是调用的时候不用而已。
下面结合*.map文件分析一下各个变量的内存分配
ma 0x2000002c Data 8 main.o(.data)
a2 0x20000034 Data 1 main.o(.data)
a3 0x20000035 Data 1 main.o(.data)
a4 0x20000036 Data 1 main.o(.data)
a5 0x20000038 Data 4 main.o(.data)
上面是各个变量的地址分配,我没有在*. map文件中找到 ma.a0 和 ma.a1 的变量地址,(~ ̄▽ ̄~) 那又怎样,不就绕个弯嘛。
<( ̄ c ̄)y▂ξ,现在可以分析了。
在上图中我通过一个指针变量将ma.a1的地址取了出来,然后执行到红点处停止,结合*.map文件分析,a2地址为0x20000034,ma地址为0x2000002c ,两者相减为8,就是ma的内存分配大小,然后我之前说过,结构体中变量的内存的分配是直接每个变量分配一个所有元素中最大的内存占用元素的,也就是说ma .a0也占用了4个字节。
现在,验证方法就是用ma.a1的地址减去ma的地址。
即 0x20000030 - 0x2000002c = 4;
结果出来了,a0变量占用了4个字节的内存,还有两个字节的内存空间去哪里了?直接屏蔽了呗。
下面继续验证我的说法,编译器给结构体分配内存是,直接分配所含成员中,数据类型最大那个的占用字节数乘以元素个数。
这张图片验证了,结构体可以分配1个字节,而结构体成员里面就一个char型变量,字节数最大占用为1。
看了这么多,下面来个题目。
struct a
{
char a0;
char b;
int a1;
}ma;
char a2,a3,a4;
int* a5;
int main()
{
a2 = sizeof(ma);
a3 = sizeof(char);
a4 = sizeof(int);
a5 = &ma.a1;
}
设ma地址0x2000002c,请问,a2、a3、a4、a5的值各为多少?
嘿嘿,是8,就是说,如果是连续的char型变量,编译器依旧会给它分配两个字节的内存空间。
把上题目改一下哈
struct a
{
char a0;
int a1;
char b;
}ma;
char a2,a3,a4;
int* a5;
int main()
{
a2 = sizeof(ma);
a3 = sizeof(char);
a4 = sizeof(int);
a5 = &ma.a1;
}
设ma地址为0x2000010c,请问,a2、a3、a4、a5的值各为多少?
这次的内存地址分配变更了,map文件如下
a2 0x2000002c Data 1 main.o(.data)a3 0x2000002d Data 1 main.o(.data)
a4 0x2000002e Data 1 main.o(.data)
a5 0x20000030 Data 4 main.o(.data)
__stdout 0x20000038 Data 4 stdout.o(.data)
Uart 0x2000003c Data 208 main.o(.bss)
ma 0x2000010c Data 12 main.o(.bss)
从上面的仿真分析,ma.a0依旧是占用4个内存字节,ma.b也是4个。
为了使验证更使人信服,我增加了两个测试,增加位定义。
如图,即使我只声明了一个位,依旧分配了4个字节。
*.map文件中的地址分配如下
ma 0x2000002c Data 2 main.o(.data)
a2 0x2000002e Data 1 main.o(.data)
a3 0x2000002f Data 1 main.o(.data)
a4 0x20000030 Data 1 main.o(.data)
a5 0x20000034 Data 4 main.o(.data)
就是说,如果连续定义8个位,只占用了两个字节
综上所述,如果可能的话,在编写c语言的时候,尽量将数据类型相同的变量连续排在一起。
写完了,看懂了吗,不懂?我就喜欢你懵逼的样子,<( ̄︶ ̄)> 。
写于2016年10月10日凌晨 深圳
- C语言好难——结构体的内存分配
- C语言好难——内存分配概述
- C/C++ 语言中结构体的内存分配
- C/C++ 语言中结构体的内存分配
- C/C++ 语言中结构体的内存分配
- C/C++ 语言中结构体的内存分配
- C/C++ 语言中结构体的内存分配
- c语言中:结构体的内存分配
- C语言结构体分配内存问题
- C语言结构体成员内存分配
- C 结构体 内存分配
- C语言——内存分配的方式
- C语言——内存分配的方式
- C语言—常见的内存分配函数
- C语言的内存分配
- C语言的内存分配
- c语言的内存分配
- C语言的内存分配
- Linux-diff和diff3命令
- android.content.res.Resources$NotFoundException:Resource ID #0x7f080004 type #0x12 is not valid问题解决
- jquery队列控制ajax执行顺序
- Fragment挂载进Activity
- 支持向量机通俗导论(一)
- C语言好难——结构体的内存分配
- Hibernate的dynamic-insert和dynamic-update的使用
- Chapter 3 Vector类还需要同步吗
- 文章标题
- java原生http请求post
- xfce4-terminal使用solarized theme
- libevent学习笔记【使用篇】——4. 让events正常工作
- 获取一个字符串里面的第一个不是重复的元素
- eclipse+Axis2+webservice之初见