一.先看一下内存图
二. 内存对⻬的原则
1:数据成员对⻬规则:结构(struct)(或联合(union))的数据成员,第
一个数据成员放在offset为0的地方,以后每个数据成员存储的起始位置要
从该成员大小或者成员的子成员大小(只要该成员有子成员,比如说是数组,
结构体等)的整数倍开始(比如int为4字节,则要从4的整数倍地址开始存
储。 min(当前开始的位置mn)m=9n=4
9 10 11 12
2:结构体作为成员:如果一个结构里有某些结构体成员,则结构体成员要从
其内部最大元素大小的整数倍地址开始存储.(struct a里存有struct b,b
里有char,int ,double等元素,那b应该从8的整数倍开始存储.)
3:收尾工作:结构体的总大小,也就是sizeof的结果,.必须是其内部最大成员的整数倍.不足的要补⻬。
三.自定义结构体
1.结构体例1:
struct LGStruct1 {
double a; //8 [0~7]
char b; // 1 8是1的倍数 [8]
int c; //4 9不是4的倍数 往后偏移到4的倍数 12,[12 13 14 15]
short d; //2 16是2的倍数 [16 17]
/*
17不是 8的倍数(double a; //8 [0~7])所以要往后偏移到24
*/
}struct1;
为什么要字节对齐
需要字节对齐的根本原因在于CPU访问数据的效率问题。例如,假设一个处理器总是从存储器中取出8个字节,则地址必须为8的倍数。如果我们能保证将所有的double类型数据的地址对齐成8的倍数,那么就可以用一个存储器操作来读或者写值了。否则,我们可能需要执行两次存储器访问,因为对象可能被分放在两个8字节存储块中。也就利用空间换取时间.
2.结构体例2:
struct LGStruct1 {
double a; //8 [0~7]
char b; // 1 8是1的倍数 [8]
int c; //4 9不是4的倍数
short d; //2
}struct1;
struct LGStruct2 {
double a; //8
int c; //4
char b; // 1
short d; //2
}struct2;
//打印
int main(int argc, char * argv[]) {
NSString * appDelegateClassName;
@autoreleasepool {
// Setup code that might create autoreleased objects goes here.
appDelegateClassName = NSStringFromClass([AppDelegate class]);
NSLog(@"输出:%lu-%lu",sizeof(struct11),sizeof(struct21));
}
return UIApplicationMain(argc, argv, nil, appDelegateClassName);
}
2021-06-09 15:52:17.358548+0800 001-内存对齐原则[2484:476177] 输出:24-16
3.结构体例3:结构体中包含结构体
①.结构体中包含结构体
struct LGStruct2 {
double a; //8 [0~7]
}struct2;
struct LGStruct1 {
char b; //1 [0]
struct LGStruct2 str; //8
/*
str 之前推算出占用 1个字节
struct LGStruct2 str 占用8个, 所以1+8 =9个;
最大的字节是LGStruct2 double a,8个字节,所以必须是它的倍数,总共占用16个字节.
*/
}struct1;
//打印
NSLog(@"输出:%lu-%lu",sizeof(struct1),sizeof(struct2));
输出:16-8
②.结构体中包含结构体
struct LGStruct2 {
char b; //1
}struct2;
struct LGStruct1 {
double a; //8 [0~7]
struct LGStruct2 str; //1
/*
str 之前推算出占用 8个字节
struct LGStruct2 str 占用1个, 所以1+8 =9个;
最大的字节是LGStruct1 double a,8个字节,所以必须是它的倍数,总共占用16个字节.
*/
}struct1;
//打印
NSLog(@"输出:%lu-%lu",sizeof(struct1),sizeof(struct2));
输出:16-1
③.结构体中包含结构体
struct LGStruct2 {
double a; //8 [0~7]
}struct2;
struct LGStruct1 {
double a; //8 [0~7]
struct LGStruct2 str; //8
/*
str 之前推算出占用 8个字节
struct LGStruct2 str 占用8个, 所以8+8 =16个;
最大的字节是LGStruct1 double a,8个字节,所以必须是它的倍数,总共占用16个字节.
*/
}struct1;
//打印
NSLog(@"输出:%lu-%lu",sizeof(struct1),sizeof(struct2));
输出:16-8
④.结构体中包含结构体
struct LGStruct1 {
double a; //8 [0~7]
char b; // 1 [8]
int c; //4 (9 10 11 [12 13 14 15]
short d; //2 [16 17] 24
}struct1;
struct LGStruct3 {
double a; //8 [0~7]
int b; //4 [8 9 10 11 12]
char c; //1 [13]
short d; //2 [14 15]
int e; //4 [16 17 18 19 20] 24
struct LGStruct1 str; //[24]
/*
str 之前推算出占用 24个字节
struct LGStruct2 str 占用24个, 所以24+24 =48个;
最大的字节是LGStruct3 double a和LGStruct1 double a都是8个字节,所以总共占用48个字节.
*/
}struct3;
//打印
NSLog(@"输出:%lu-%lu",sizeof(struct3),sizeof(struct1));
输出:48-24
未完待续