C/C++内存对齐规则(结构体、联合体、类)

简介: C/C++内存对齐规则(结构体、联合体、类)

前言


求结构体的大小是很热门的考点,无论你是学C还是C++,都会遇到这样的问题,在面试中也很受欢迎,所以我们先思考这样一个问题:计算结构体,联合体和类的大小应该怎么去计算呢?我们知道,在C语言中结构体内部可以包含很多变量,所以我们在没有学习这个知识之前,会固化地认为,结构体的大小就是内部变量的大小的总和,但是事实就是这样吗?显然并不是,那接下来我们就要去学习一下如何求结构体它们的大小

一、内存对齐规则(每一个都是重点

1. 第一个成员永远在与结构体变量偏移量为0的地址处。


2. 其他成员变量要对齐到 自身对齐数的整数倍 的地址处。


对齐数 = 编译器默认的一个对齐数 与 该成员大小的较小值。


只有VS中默认对齐数值为8

其他编辑器(gcc,clang)的对齐数就是成员自身的大小


3. 结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍。


4. 如果一个结构体的成员有另一个结构体类型的变量,这个结构体类型的变量要对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数的整数倍。


5. 当结构体里有数组的时候,我们数组的对齐数是数组中一个元素的大小,之后在占整个数组大小的空间。5. 使用下面的代码可以设置默认对齐数:

#include <stdio.h>
#pragma pack(8)//设置默认对齐数为8
struct S1
{
    char c1;
    int i;
    char c2;
};
#pragma pack()//取消设置的默认对齐数,还原为默认


二、求结构体大小的例题 (都按编辑器没有默认对齐数)

1. 例题1

#include <stdio.h>
struct S1
{
    char c1;
    int i;
    char c2;
};
int main()
{
    printf("%lu",sizeof(struct S1));
    return 0;
}

答案及解析 12

2. 例题2

#include <stdio.h>
struct S2
{
    char c1;
    char c2;
    int i;
};
int main()
{
    printf("%lu\n", sizeof(struct S2));
    return 0;
}


答案及解析 8

153. 例题3

#include <stdio.h>
struct S3
{
    double d;
    char c;
    int i;
};
int main()
{
    printf("%lu\n", sizeof(struct S3));
    return 0;
}


答案及解析 16

4. 例题4 (嵌套结构体)

#include <stdio.h>
struct S3
{
    double d;
    char c;
    int i;
};
struct S4
{
    char c1;
    struct S3 s3;
    double d;
};
int main()
{
    printf("%lu\n", sizeof(struct S4));
    return 0;
}


答案及解析 32

5. 例题5(结构体内部有数组)

#include <stdio.h>
struct S1
{
    char c1;
    int arr[6];
};
int main()
{
    printf("%lu",sizeof(struct S1));
    return 0;
}


答案及解析 28

三、联合体的计算规则


联合体是共用同一块空间的;

1. 联合的大小至少是最大成员的大小

2. 当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍

3. 数组的对齐数是数组的一个元素的大小,再占整个数组大小的空间。

四、求union联合体大小的例题

1. 例题1

#include <stdio.h>
union Un1
{
    char c[5];
    int i;
};
int main()
{
    printf("%lu\n", sizeof(union Un1));
    return 0;
}

答案及解析  8

2. 例题2

#include <stdio.h>
union Un2
{
    short c[7];
    int i;
};
int main()
{
    printf("%lu\n", sizeof(union Un2));    
    return 0;
}

答案及解析 16

五、求类大小


向结构体看齐,一样的规则哦!但是有几点不跟结构体一样;


1. 类的内部有成员变量和成员函数,我们只算成员变量的就行,不需要算成员函数;


2. 不包含成员变量的类默认的大小是1字节,空类也是1字节


3. 静态成员变量不属于实例化对象,所以不会占类的空间,因为静态成员变量是一种特殊的静态全局变量,写在类里是为了表明它在哪个作用域而已。


4. 类里面包含类,计算大小的时候,不需要算类里面的类,内部的类其是跟我们外部的类平行的关系,放在类里面只是为了在访问的时候,需要指明在哪个作用域里。

1. 例题1

#include <iostream>
using namespace std;
class A
{
public:
    void Print()
    {
        cout << "Print()" << endl;
    }
private:
};
int main()
{
    cout << sizeof(A) << endl;
    return 0;
}


答案及解析 1

没有成员变量的类大小为1个字节;

2. 例题2

#include <iostream>
using namespace std;
class A
{
public:
    void Print()
    {
        cout << "Print()" << endl;
    }
private:
    double _a;
    char _b;
    int _c;
};
int main()
{
    cout << sizeof(A) << endl;
    return 0;
}

答案及解析 16

算类的大小,只需要算成员变量的大小,不需要算成员函数,因为成员函数属于类的共有部分,每个实例化对象都有这个函数,难bfb411d05853410ea73970360a138f81.png道我们每次实例化一次对象,就要创造成员函数吗?很显然不是,因为太消耗空间里,而每个实例化对象都必须有属于自己的成员变量,这是毋庸置疑的;




3. 例题3 (静态成员变量)

#include <iostream>
using namespace std;
class A
{
public:
    void Print()
    {
        cout << "Print()" << endl;
    }
private:
    double _a;
    static int _b;
};
int main()
{
    cout << sizeof(A) << endl;
    return 0;
}


答案及解析 8

静态成员变量是所有类的实例化对象共享的,并不是每个对象都要创建的,所以跟成员函数一个道理,不用算

4. 例题4 (类类型的成员变量)

#include <iostream>
using namespace std;
class A
{
public:
    void Print()
    {
        cout << "Print()" << endl;
    }
private:
    double _a;
};
class B
{
private:
    int _b;
    A _a;
};
int main()
{
    cout << sizeof(B) << endl;
    return 0;
}


答案及解析 16

要注意的是这不是内部类,是在成员变量里有一个A类类型的变量,这就跟结构体里面有另外一个结构体变量一样,如果一个结构体的成员有另一个结构体类型的变量,这个结构体类型的变量要对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数的整数倍。

b11eae4d11764dd1bb23bc137c987c13.png

5. 例题5 (内部类)

#include <iostream>
using namespace std;
class A
{
public:
    void Print()
    {
        cout << "Print()" << endl;
    }
private:
    double _a;
    class B
    {
    private:
        int _b;
    };
};

答案及解析 8

内部类不需要算,因为是与外部类平行的关系

相关文章
|
28天前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
54 12
|
2月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
58 16
|
2月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
2月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
2月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
148 6
|
3月前
|
存储 算法 Java
JVM: 内存、类与垃圾
分代收集算法将内存分为新生代和老年代,分别使用不同的垃圾回收算法。新生代对象使用复制算法,老年代对象使用标记-清除或标记-整理算法。
47 6
|
2月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
2月前
|
Java
非静态内部类持有外部类引用导致内存溢出
非静态内部类持有外部类引用导致内存溢出
|
3月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
3月前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)

热门文章

最新文章