关于结构体中变量空间占用问题思考

面试C技术的人在基础笔试时总会被问的问题是变量空间的问题

尤其是一些混合结构的类型空间

本次就对其中具有代表性的结构体类型进行说明,便于给长记性

测试环境: gcc 4.8.1 Ubuntu 13.04 32-bit

就从以下的实例开始说起:

struct Pad1{
    char a;
};

struct Pad2{
    int b;
};

struct Pad3{
    char a;
    int b:2;
    double c;
};

struct Pad4{
    char a;
    double c;
    int b:2;
};

以上定义了4个结构类型,采用基本的数据类型定义(复杂类型在这个基础上可以扩展)

提出问题:

1、以上结构体的大小在32位系统和64位系统的结果是否一致?

2、大小端存储类型是否有对以上结果是否有影响?

3、结构体内变量顺序是否对结构体空间大小有影响?有无最小化规律可循?

4、指针变量类型对结构体有什么影响?

分析问题:

根据实践,以上结构体的空间大小(单位:byte)结果为:

sizeof Pad1 is: 1
sizeof Pad2 is: 4
sizeof Pad3 is: 12
sizeof Pad4 is: 16

根据以上内容,看到这样的结果最令人奇怪的是两个方面:

A、Pad3与Pad4变量内容相同(顺序差异),然而结果却不同

B、定义int b:2 的目的是什么,如果仅定义 int 会有什么不同

咱们先来说B,定义之所以定义限定位数的整型变量,是因为如果不这样写,两个sizeof的结果相同!

这就引出了A,当结构体中的变量内容相同,仅顺序不同会有什么什么影响

回答这个问题需要了解一个前提:内存存储都是整数对齐的,即按照字节或者字对齐

因此大多数编译器为了迎合这个对齐的功能,在进行编译时会将变量进行对齐处理

在32bit系统中,按照4个字节进行对齐,因此当出现混合类型的变量时

sizeof的结果不是简单的累加,而要考虑对齐的扩展,可以确定的是不比理论值小。

至于大了多少,要看那些变量类型涉及到对齐,以Pad3和Pad4为例

理论上都是  total =   a:1 + b:2 + c:8 = 11

Pad3中a和b连续,加起来不足4个字节,因此为了对齐将两者共分配4个字节,

最终结果为啥是12(4+8)就不难理解了;

同理,Pad4中因为a与c是连续的,两者加起来是9 不足4的倍数,补足后是12

加上最后的b(2)是14,再补得16。

解决问题:

通过以上分析,回答提出的几个问题:

1、32位与64位的差别:就是对齐字节数不同,32位是4个字节,64位是8个字节

所以两者结果有是否会有不同,答案可以自己实践一下。

2、大小端类型(不明白这个名词的可以搜索一下)对这个所占空间结果没有影响

但对联合体中变量的赋值会有影响,所以这里特别指出。

3、变量顺序是否有影响已经在示例中说明了,肯定的说是有的

但是正如例子中说的之所以有的时候没有影响,那只是因为前后结合后造成的补齐结果相同而已

例如: 1+ 4 + 8 结合顺序为 5 --》 8  + 8 = 16 ,如果是 1 + 8 + 4 结合顺序为 9 --》 12 + 4 = 16

虽然都是16 ,但本质是有差别的。

要使空间最小化,一个常用的参考就是哈夫曼原则,即从小到大进行排序定义变量。

4、指针类型变量记住一个原则:大小永远是4个字节(64位是8个字节)



### 回答1: 结构体类型本身不占用内存空间,只有在定义结构体变量时才会分配内存空间结构体变量在内存占用空间大小取决于结构体成员的数据类型和对齐方式,即内存对齐。在结构体定义,可以使用特殊的指令来指定结构体成员的对齐方式,以优化内存空间的利用。 ### 回答2: 结构体是一种自定义数据类型,可以由多个不同数据类型的变量组合而成。结构体类型本身不占用内存空间,它只是用来定义一个数据结构的模板,描述了结构体包含的变量的类型和顺序。 结构体变量是根据结构体类型定义的变量,它占用内存空间。当定义一个结构体变量时,系统会根据结构体类型定义的变量类型和顺序来分配内存空间,并为每个变量分配对应的内存。 例如,定义一个包含姓名和年龄的结构体类型: ```c struct Person { char name[20]; int age; }; int main() { struct Person p1; // 定义一个结构体变量p1 printf("sizeof(struct Person): %d\n", sizeof(struct Person)); // 结果为24,表示结构体变量需要占用24个字节的内存空间 return 0; } ``` 在上面的例子结构体类型`struct Person`本身不占用内存空间,而结构体变量`p1`占用了24个字节的内存空间,其包括了`name`变量的20个字节和`age`变量的4个字节。 结构体变量的内存空间可以通过`sizeof`运算符来获取,它返回的是结构体变量在内存占用的字节数。这个大小是由结构体变量类型和对齐方式决定的。 总结起来,结构体类型本身不占用内存空间,只有结构体变量才会占用内存空间结构体类型用于定义结构体变量的类型,描述其包含的变量信息。 ### 回答3: 结构体类型是一种用户自定义的数据类型,用于将多个不同类型的数据组合在一起,构成一个新的数据类型。在内存结构体类型本身不占用任何空间,它只是一个用来描述数据组织方式的模板。 然而,当我们定义一个结构体变量时,实际上会在内存分配一块用于存储该变量数据的空间。这个空间的大小取决于结构体各个成员的大小和对齐规则。 结构体的成员按照定义的顺序在内存依次排列。如果其有成员的类型是基本数据类型,其大小就是该类型所占用的字节数。如果有成员的类型是数组、指针或其他结构体类型,那么结构体的大小将包含这些成员所占用空间。 此外,编译器会在结构体的成员之间插入一些字节,以满足对齐规则。对齐是为了提高访问成员变量的效率。例如,某个平台的对齐规则可能要求成员变量按照4字节对齐。如果成员变量的大小是4的倍数,则其起始地址就满足对齐要求;否则,编译器会在前面的成员变量之后插入一些字节,使得下一个成员变量的起始地址满足对齐要求。 总之,结构体类型本身不占用内存空间,仅仅是一个描述数据组织方式的模板。而结构体变量在内存占用空间,其大小包括了所有成员变量的大小以及对齐字节。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值