当前位置:首页 > 芯闻号 > 充电吧
[导读]struct的对齐问题是由一道笔试题想到的,笔试题如下:#includeusing namespace std; typedef union student {     char name[10];

struct的对齐问题是由一道笔试题想到的,笔试题如下:

#includeusing namespace std;
typedef union student
{
    char name[10];
    long sno;
    char sex;
    float score [4];
} STU;
void main ()
{
    STU a[5];
    cout << sizeof(a) <<endl;//输出:80
}

答案是80,因为union是可变的以其成员中最大的成员作为该union的大小!
但是换成是struct又是多少呢?

#includeusing namespace std;
typedef struct student
{
    char name[10];
    long sno;
    char sex;
    float score [4];
} STU;
void main ()
{
    STU a[5];
    cout << sizeof(a) <<endl;//输出:180
}

答案是180?为什么不是(10+4+1+16)*5=155?因为struct有个叫对齐方式的问题:
一.自然对齐
不对齐的数据存取在x86上影响速度,对齐即是多分配一些字节,填充无用数据,以空间的损失来换取效率。
struct是一种复合数据类型,其构成元素既可以是基本数据类型(如int、long、float等)的变量,也可以是一些复合数据类型(如array、struct、union等)的数据单元。对于结构体,编译器会自动进行成员变量的对齐,以提高运算效率。缺省情况下,编译器为结构体的每个成员按其自然对齐(natural alignment)条件分配空间。各个成员按照它们被声明的顺序在内存中顺序存储,第一个成员的地址和整个结构的地址相同。
自然对齐(natural alignment)即默认对齐方式,是指按结构体的成员中(类型)size最大的成员作为基本的分配单元,而且与其顺序有这密切的联系。
例如:

#includeusing namespace std;
struct naturalalign
{
 char a;
 short b;
 char c;
};
void main ()
{
    cout << sizeof(naturalalign ) <<endl;//输出:6
}

在上述结构体中,size最大的是short,其长度为2字节,因而结构体中的char成员a、c都以2为单位对齐,sizeof(naturalalign)的结果等于6。
如果改为:

#includeusing namespace std;
struct naturalalign
{
 char a;
 int b;
 char c;
};
void main ()
{
    cout << sizeof(naturalalign) <<endl;//输出:12
}

其结果显然为12。
那么再回到到原题:结构体中,size最大的是long,size是4,所以,按照顺序,char name[10],12个字节;long sno,4个字节;char sex,4个字节(这里对齐了);float score [4],16个字节。于是(12+4+4+16)×5=180,就是了!
刚才还说过,与顺序有关,我们改一下:

#includeusing namespace std;
typedef struct student
{
     char name[10];
     char sex;
     long sno;
     float score [4];
} STU;
void main ()
{
    STU a[5];
    cout<<sizeof(a)<<endl;//输出:160
}

答案是160。为什么,只是换了顺序而已呀?关键就在顺序上。
结构体中,size最大的是long,size是4,所以,按照顺序,char name[10],12个字节;但是这12中多分配的2个字节可以包含后面的char sex(问题就在这);float score [4],16个字节。于是(12+4+16)×5=160,就是了!所以要小心呀!
二.指定对齐
一般地,可以通过下面的方法来改变缺省的对齐条件:
☆使用伪指令#pragma pack (n),编译器将按照n个字节对齐;
☆使用伪指令#pragma pack (),取消自定义字节对齐方式。
注意:如果#pragma pack (n)中指定的n大于结构体中最大成员(类型)的size,则其不起作用,结构体仍然按照size最大的成员进行对齐。
例如:

#includeusing namespace std;
#pragma pack (n)
struct naturalalign
{
    char a;
    int b;
    char c;
};
#pragma pack()
void main ()
{
    cout<<sizeof(naturalalign)<<endl;
}

当n为4、8、16时,其对齐方式均一样,sizeof(naturalalign)的结果都等于12。而当n为2时,其发挥了作用,使得sizeof(naturalalign)的结果为8。
再看一个例子:

#pragma pack(2)  
struct test     //test采用2字节对齐 
{  
    char c1;    
    short s;     
    int i;      
    char c2;     
    float f;    //f长度4字节  
    double d;   //d长度8字节  
    char c3;    
};  
#pragma pack()  
void main ()
{
    cout<<sizeof(test)<<endl;//输出:24
}

test的内存分布如下(以1表示占用字节,*表示空字节)

c1 s i c2 f d c3 1* 11 1111 1* 1111 11111111 1*

在VC++ 6.0编译器中,我们可以指定其对齐方式,其操作方式为依次选择projetct > setting > C/C++菜单,在struct member alignment中指定你要的对界方式。


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

其实在 c++语言里面const修饰的才算是一个真正的常量,在 c 语言中 const 可以说是个“冒牌货”。为什么会这样?其实是 c++ 编译器对 const 进行了加强,当 c++ 编译器遇到常量声明时,不会像 c...

关键字: c++ C语言 const

返回函数的引用去初始化一个新的引用这个和前面一样,都是不会产生副本,但是现在是用返回值去初始化一个引用声明c,也就是说这时候变成了变量temp的别名,在c的生命周期内temp是一直有效的,这样做完全可以。

关键字: c++ 返回值 引用声明

C++是一种面向对象的高级程序设计语言,是C语言的超集。

关键字: c++ C语言

分析:这是Adobe 公司2007 年校园招聘的最新笔试题。这道题除了考察应聘者的C++ 基本功底外,还能考察反应能力,是一道很好的题目。 在Java 中定义了关键字final ,被final 修饰的

关键字: c++ class

泛型算法中的定制操作很多算法都会比较输入序列中的元素,通过定制比较动作,可以控制算法按照编程者的意图工作。本文以string排序为例进行说明,首先是缺省的排序动作: vector v{"This","

关键字: c++

为什么是lambda?讲了这么多天的lambda表达式,有一个很基本的问题没有回答:为什么叫lambda表达式呢?首先这个lambda就是罗马字母λ,lambda表达式即λ表达式。数学上有一个概念叫λ

关键字: c++

        假设我们有个函数用来揭示处理程序的优先权,另一个函数用来在某动态分配所得的Widget 上进行某些带有优先权的处理:int priority () ; void processWi

关键字: c++ effective

判断链表中是否有环最经典的方法就是快慢指针,同时也是面试官大多想要得到的答案。       快指针pf(f就是fast的缩写)每次移动2个节点,慢指针ps(s为slow的缩写)每次移动1个节点,如果快

关键字: c++ 链表 快慢指针

ok6410 linux的第一个驱动LED驱动 //驱动程序代码/*******************************************************************

关键字: Linux struct

这个驱动人家都说很简单,因为需要,所以我自己写了一个。 其实最后发现要是初学者写这个驱动还是不简单的,问了好多人,发了好多帖子,会的人寥寥无几啊,网上贴的都是一些只有中断输入button和只有输出的l

关键字: output struct
关闭
关闭