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

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

相关文章
|
8月前
|
Arthas 存储 算法
深入理解JVM,包含字节码文件,内存结构,垃圾回收,类的声明周期,类加载器
JVM全称是Java Virtual Machine-Java虚拟机JVM作用:本质上是一个运行在计算机上的程序,职责是运行Java字节码文件,编译为机器码交由计算机运行类的生命周期概述:类的生命周期描述了一个类加载,使用,卸载的整个过类的生命周期阶段:类的声明周期主要分为五个阶段:加载->连接->初始化->使用->卸载,其中连接中分为三个小阶段验证->准备->解析类加载器的定义:JVM提供类加载器给Java程序去获取类和接口字节码数据类加载器的作用:类加载器接受字节码文件。
767 55
|
6月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
233 26
|
11月前
|
存储 程序员 编译器
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
|
7月前
|
C语言 C++
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
137 1
|
存储 缓存 编译器
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
542 68
|
10月前
|
存储 Linux C语言
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
|
10月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
537 6
|
11月前
|
存储 算法 Java
JVM: 内存、类与垃圾
分代收集算法将内存分为新生代和老年代,分别使用不同的垃圾回收算法。新生代对象使用复制算法,老年代对象使用标记-清除或标记-整理算法。
171 6
|
10月前
|
Java
非静态内部类持有外部类引用导致内存溢出
非静态内部类持有外部类引用导致内存溢出
|
11月前
|
安全 C语言 C++
彻底摘明白 C++ 的动态内存分配原理
大家好,我是V哥。C++的动态内存分配允许程序在运行时请求和释放内存,主要通过`new`/`delete`(用于对象)及`malloc`/`calloc`/`realloc`/`free`(继承自C语言)实现。`new`分配并初始化对象内存,`delete`释放并调用析构函数;而`malloc`等函数仅处理裸内存,不涉及构造与析构。掌握这些可有效管理内存,避免泄漏和悬空指针问题。智能指针如`std::unique_ptr`和`std::shared_ptr`能自动管理内存,确保异常安全。关注威哥爱编程,了解更多全栈开发技巧。 先赞再看后评论,腰缠万贯财进门。
492 0