下面这个篇博客讲解很好

http://blog.csdn.net/meegomeego/article/details/9393783

总的来看分三类:

1. 不加 #pragma pack(n)伪指令(n未对齐的字节数目)

2. 加#pragma pack(n) 伪指令

3. __attribute__((aligned(n)))来使用n个字节方式的对齐

第一种:不加 #pragma pack(n)伪指令

对其的字节数目为一个结构体中的基本类型(int char float double,特别强调复合型(数组、结构体、联合体)不算。)成员变量中最大的字节数(也可理解为默认对齐字节数,如char 默认 1字节 int 默认 4字节 等)。

第二种:加#pragma pack(n) 伪指令

对其的字节数目为一个结构体中的基本类型(int char float double,特别强调复合型(数组、结构体、联合体)不算。)成员变量最大的字节数(也可理解为默认对齐字节数,如char 默认 1字节 int 默认 4字节 等) 与pragma pack(n) 中的n值二者之间的较小数为对齐的字节数。比如:一个结构体中基本类型的

成员变量为 float a;//默认是4字节对齐,而加#pragma pack(8) 设置为8字节,此时加#pragma pack(8)设置失效,是4字节对齐(4 < 8),如果 n 为 2,那么对齐的字节数就是2字节了(2 < 4)。

经典例子:

 1 // 例3(如果在GCC中编译,请加上-malign-double选项)
2 #pragma pack(8)
3 struct s1
4 {
5 short a; // 第一个成员,放在[0, 1]偏移的位置。
6 long b; // 第二个成员sizeof(long)=4, #pragma pack(8), 取小值也就是4
7 // 所以这个成员按4字节对齐,放在偏移[4~7]的位置。
8 };
9 // struct s1中size最大的数据成员(4), #pragma pack(8)
10 // 取小值也就是4,所以sizeof(struct s1)应当按照4来对齐,为8
11
12 struct s2
13 {
14 char c; // 第一个成员,放在[0]偏移的位置。
15 struct s1 d; // 第二个成员为struct s1,其对齐方式是它的所有成员使用的对齐参数中最大的一个,即4。
16 // 所以第二个成员d按4字节对齐,由于sizeof(d)=8, 放在偏移[4~11]的位置。
17 long long e; // 第三个成员sizeof(long long)=8, #pragma pack(8), 取小值也就是8
18 // 所以这个成员按8字节对齐,放在偏移[16~23]的位置。
19 };
20 #pragma pack()

第三种:__attribute__((aligned(n)))来使用n个字节方式的对齐

这种方式和第二种的方式是刚好相反的,取得是较大值来作为字节对齐。

经典例子:

 1 #include <stdio.h>
2 #include <stdlib.h>
3
4 #pragma pack(16)
5 struct testa
6 {
7 unsigned int number;
8 };
9 #pragma pack()
10 struct testa arr_testa[4] = { {65535}, {65535}, {65535}, {65535} };
11
12 struct testb
13 {
14 unsigned int number;
15 } __attribute__ ((aligned (16))) ;
16
17 struct testb arr_testb[4] = { {65535}, {65535}, {65535}, {65535} };
18
19
20 int main(int argc, const char *argv[])
21 {
22 printf("sizeof(struct testa) = %d, sizeof(arr_testa) = %d\n",
23 sizeof(struct testa), sizeof(arr_testa));
24 printf("sizeof(struct testb) = %d, sizeof(arr_testb) = %d\n",
25 sizeof(struct testb), sizeof(arr_testb));
26
27 printf("address of arr_testa[0] = %p\n", &arr_testa[0]);
28 printf("address of arr_testa[1] = %p\n", &arr_testa[1]);
29 printf("address of arr_testa[2] = %p\n", &arr_testa[2]);
30 printf("address of arr_testa[3] = %p\n", &arr_testa[3]);
31 puts("");
32 printf("address of arr_testb[0] = %p\n", &arr_testb[0]);
33 printf("address of arr_testb[1] = %p\n", &arr_testb[1]);
34 printf("address of arr_testb[2] = %p\n", &arr_testb[2]);
35 printf("address of arr_testb[3] = %p\n", &arr_testb[3]);
36 return 0;
37 }

输出结果为:

 1 sizeof(struct testa) = 4, sizeof(arr_testa) = 16
2 sizeof(struct testb) = 16, sizeof(arr_testb) = 64
3 address of arr_testa[0] = 0x804a040
4 address of arr_testa[1] = 0x804a044
5 address of arr_testa[2] = 0x804a048
6 address of arr_testa[3] = 0x804a04c
7
8 address of arr_testb[0] = 0x804a060
9 address of arr_testb[1] = 0x804a070
10 address of arr_testb[2] = 0x804a080
11 address of arr_testb[3] = 0x804a090

 特别强调注意union 的字节对齐:

1 union AAA{
2   double i;//8B
3 int k[5];//20B
4 cahr b; //1B
5 }

上面的unionAAA类型,类型的字节大小为sizeof(AAA) ,按固有的原则来看,应该取AAA中长度最长的成员的程度。但是容易忽略一点就是字节对齐问题。综合考虑union类型的字节大小的求解,从如下两方面入手:

1. 找到占内长度最大的成员

2. 考虑字节对齐问题

union  AAA:

1. k[5] 占内存20字节最大

2. 最长的基本类型成员为double类型,占据8字节。故最后union AAA占用内存应该是8的倍数,20 + 4 即可

sizeof(AAA)  = 24字节!

C语言中的字节对齐的更多相关文章

  1. C语言中的字节对齐以及其相关处理

    首先,我们来了解下一些基本原理: 一.什么是字节对齐一个基本类型的变量在内存中占用n个字节,则该变量的起始地址必须能够被n整除,即: 存放起始地址 % n = 0,那么,就成该变量是字节对齐的;对于结 ...

  2. C语言中结构体对齐问题

    C语言中结构体对齐问题 收藏 关于C语言中的结构体对齐问题 1,比如: struct{short a1;short a2;short a3;}A;struct{long a1;short a2;}B; ...

  3. C语言中的内存对齐

    最近看了好多,也编了好多C语言的浩强哥书后的题,总觉的很不爽,真的真的好怀念linux驱动的代码,好怀念那下划线,那结构体,虽然自己还很菜. 同时看了一遍陈正冲老师的C语言深度剖析,收益很多,又把唐老 ...

  4. C语言:内存字节对齐详解[转载]

    一.什么是对齐,以及为什么要对齐: 1. 现代计算机中内存空间都是按照byte划分的,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但实际情况是在访问特定变量的时候经常在特定的内存地址访问, ...

  5. C语言:内存字节对齐详解

    转:http://blog.csdn.net/arethe/article/details/2548867 一.什么是对齐,以及为什么要对齐: 1. 现代计算机中内存空间都是按照byte划分的,从理论 ...

  6. ACE的CDR中的字节对齐问题

    大家应该都知道计算机中间都有字节对齐问题.CPU访问内存的时候,如果从特定的地址开始访问一般可以加快速度,比如在32位机器上,如果一个32位的整数被放在能被32模除等于0的地址上,只需要访问一次,而如 ...

  7. c语言,内存字节对齐

    引用:内存字节对齐 写出一个struct,然后sizeof,你会不会经常对结果感到奇怪?sizeof的结果往往都比你声明的变量总长度要大,这是怎么回事呢?讲讲字节对齐吧. /************* ...

  8. C++中的字节对齐分析

    struct A { int a; char b; short c; }; struct B { char a; int b; short c; }; #pragma pack(2) struct C ...

  9. [置顶] 什么是C语言结构体字节对齐,为什么要对齐?

    一.概念 对齐跟数据在内存中的位置有关.如果一个变量的内存地址正好位于它长度的整数倍,他就被称做自然对齐.比如在32位cpu下,假设一个整型变量的地址为0x00000004,那它就是自然对齐的.   ...

随机推荐

  1. Python之@property详解及底层实现介绍

    转自:https://blog.csdn.net/weixin_42681866/article/details/83376484 前文 Python内置有三大装饰器:@staticmethod(静态 ...

  2. Machine learning(3-Linear Algebra Review )

    1.Matrices and vectors Matrix :Rectangular array of numbers a notation R3×3 Vector : An n×1 matrix t ...

  3. Qt信号与槽传递自定义数据类型——两种解决方法

    信号与槽作为qt中的核心机制,在qt应用开发中经常会用的,但是原生的信号与槽连接传参,只支持基本的数据类型,比如char,int, float,double. 如果想要在信号与槽之间传递自定义参数,比 ...

  4. 链表中倒数第K个结点 牛客网 剑指Offer

    链表中倒数第K个结点 牛客网 剑指Offer 题目描述 输入一个链表,输出该链表中倒数第k个结点. # class ListNode: # def __init__(self, x): # self. ...

  5. 数字孪生 3D 科技馆的科学传播新模式

    前言 科技馆是一种参与型体验型的博物馆,以传播科学知识.培养公众的科学创新技术为宗旨,并以其生动的展现方式得到公众的广泛欢迎.一直以来,我国科技馆的发展受到各种因素的制约和影响,发展缓慢.如今在我国经 ...

  6. python语法与pycharm的基本使用

    内容概要 pycharm基本使用 python注释语法 变量与常量 垃圾回收机制 数据类型 1. pycharm基本使用 pycharm安装完成后首次打开要注意: 文件路径(不要选择C盘) pytho ...

  7. WPF进阶技巧和实战03-控件(4-基于范围的控件及日期控件)

    系列文章链接 WPF进阶技巧和实战01-小技巧 WPF进阶技巧和实战02-布局 WPF进阶技巧和实战03-控件(1-控件及内容控件) WPF进阶技巧和实战03-控件(2-特殊容器) WPF进阶技巧和实 ...

  8. 概述C# virtual修饰符

    摘要:C#是继C++和Java语言后的又一面向对象的语言,在语法结构,C#有很多地方和C++及Java相似,但是又不同于它们,其中一些关键特别需要引起我们的注意. C# virtual修饰符用于修改方 ...

  9. OAuth 2.0 扩展协议之 PKCE

    前言 阅读本文前需要了解 OAuth 2.0 授权协议的相关内容, 可以参考我的上一篇文章 OAuth 2.0 的探险之旅. PKCE 全称是 Proof Key for Code Exchange, ...

  10. 大爽Python入门教程 1-3 简单的循环与判断

    大爽Python入门公开课教案 点击查看教程总目录 这里只初步认识下循环和判断,以便于我们去实现一些简单的计算. 循环和判断的详细知识和细节,我们将在后面的章节(大概是第三章)展开阐述. 1 初步了解 ...