【C语言】结构体内存对齐

目录

引入结构体 

 结构的声明

创建和初始化

 内部元素的使用;

特殊声明:

结构体在内存中的对齐

练习: 



引入结构体 

        C语言有各种数据类型,我们已经对一些数据类型很熟悉:

  1. 整型(int)- 存储整数值,包括有符号和无符号两种类型。

  2. 浮点型(float、double、long double)- 存储实数值,包括单精度(float)、双精度(double)和长双精度(long double)三种类型。

  3. 字符型(char)- 存储单个字符,包括有符号和无符号两种类型。

  4. 指针类型(pointer)- 存储内存地址,可以指向其他数据类型。

         但是,仅仅有这些类型是无法描述一些对象的特征的,比如:一个人的多个信息,一个家庭的多个成员等等,那么这时候,就要用到结构体类型来描述一个比较复杂的对象。

        结构体类型就像其他简单数据类型(int,float,double)一样,也是一种数据类型,但是它比较复杂,可以包含多个 简单的数据类型, 这些被包含的数据类型可以是不同的类型。

 

         总之,结构体是为了描述复杂对象而引入的一种新的数据类型。

 结构的声明

        C语言中,结构体(struct)是一种自定义的数据类型,可以将不同的数据类型组合在一起形成一个 新的 数据类型。结构体的定义格式如下:

struct 结构体名称
{
    数据类型1 成员变量1;
    数据类型2 成员变量2;
    ...
} 创建的同类型结构体名称;

         有了结构体类型,我们就可以描述复杂对象的特征。例如,要描述一个家庭的成员的信息,可以:

        创建结构体类型Person——包含变量字符串型name, 整型age, 双精度浮点型heignt;

        创建结构体类型Family——包含结构体类型Person, 字符串型location;

 代码如下:

​//声明Person结构体类型

struct Person {
    char name[20];
    int age;
    double height;
};



//声明Family结构体类型

struct Family
{
    struct Person;
    char location[10];
}Family_of_my;

​

从上述代码可以得出:

        结构体名称 根据具体的实际意义来决定。

        成员变量可以有多个,每个成员变量的数据类型可以是任意数据类型,包括简单数据类型和复杂数据类型。也就是说,一个结构体内部可以再嵌套包含一个结构体类型。

 

        总结,结构体内部的数据类型多样,可以是简单类型,也可以是复杂类型。

 

创建和初始化

        结构体的初始化可以按照不同方式进行:

        1.按照结构体内部元素顺序

        2.按照指定顺序

#include <stdio.h>
struct Stu
{
    char name[20];//名字

    int age;//年龄

    char sex[5];//性别

    char id[20];//学号

};

int main()
{

    //按照结构体成员的顺序初始化

    struct Stu s = { "张三", 20, "男", "20230818001" };

    printf("name: %s\n", s.name);
    printf("age : %d\n", s.age);
    printf("sex : %s\n", s.sex);
    printf("id : %s\n", s.id);

    //按照指定的顺序初始化

    struct Stu s2 = { .age = 18, .name = "lisi", .id = "20230818002", .sex = "⼥

    printf("name: %s\n", s2.name);
    printf("age : %d\n", s2.age);
    printf("sex : %s\n", s2.sex);
    printf("id : %s\n", s2.id);
    return 0;

}

        总结, 初始化可以按照正常顺序,还可按照指定顺序。

 内部元素的使用;

        要用到结构体操作符。 

​
​
#include <stdio.h>
struct Stu
{
    char name[20];//名字

    int age;//年龄

    char sex[5];//性别

    char id[20];//学号

};

int main()
{

    //按照结构体成员的顺序初始化

    struct Stu s = { "张三", 20, "男", "20230818001" };

    printf("name: %s\n", s.name);
    printf("age : %d\n", s.age);
    printf("sex : %s\n", s.sex);
    printf("id : %s\n", s.id);

    //按照指定的顺序初始化

    struct Stu* s2 = &s;

    printf("name: %s\n", s2->name);
    printf("age : %d\n", s2->age);
    printf("sex : %s\n", s2->sex);
    printf("id : %s\n", s2->id);
    return 0;

}

​

​

 

         总结,要使用结构体内部元素,要用到结构体操作符:

.  和 ->

用例:

结构体.成员名

结构体指针->成员名

特殊声明:

        在声明结构的时候,可以不完全的声明。也就是,可以省略结构体的标签。

例如:

//匿名结构体类型
struct
{
int a;
char b;
float c;
}x;

struct
{
int a;
char b;
float c;
}a[20], *p;

        但是,这两个匿名的类型完全一致的结构体是同一个结构体类型吗?

也就是说:

p = &x;

        这段代码合法吗?

        经过测试,编译器会把上面的两个声明当成完全不同的两个类型,所以是⾮法的。
匿名的结构体类型,如果没有对结构体类型重命名的话,基本上只能使⽤⼀次。
 

 

结构体在内存中的对齐

        现在,我们已经对结构体有了一个基本的认识,那么,结构体的大小是如何计算的呢?


 

        我们先看一个实例:

计算下列两个结构体的大小:


//练习1
struct S1
{
char c1;
int i;
char c2;
};
printf("%d\n", sizeof(struct S1));


//练习2
struct S2
{
char c1;
char c2;
int i;
};
printf("%d\n", sizeof(struct S2));

        我们可以先猜测:

        既然两个结构体内部的变量类型和数目完全一致,那么我们有理由猜测两个结构体大小相同,但是,实际情况并不是这样:

 

        其实,结构体在内存中的存储有一套自己的规则:

 首先,引入一个概念:

        对齐数

        对齐数=编译器默认的⼀个对齐数   和 该成员变量长度的较小值

        默认对齐数 :

VS 中默认的值为 8 


Linux中gcc没有默认对⻬数,对⻬数就是成员⾃⾝的⼤⼩

结构体的基本对齐方法是:

        1.结构体的第⼀个成员对⻬到和结构体变量起始位置偏移量为0的地址处
        2.其他成员变量要对齐到对齐数的整数倍的地址处

结构体大小的计算:


        1.结构体总⼤⼩为最⼤对⻬数(结构体中每个成员变量都有⼀个对⻬数,所有对⻬数中最⼤的)的整数倍。


        2.如果嵌套了结构体的情况,嵌套的结构体成员对⻬到⾃⼰的成员中最⼤对⻬数的整数倍处,结构体的整体⼤⼩就是所有最⼤对⻬数(含嵌套结构体中成员的对⻬数)的整数倍。

 

        这样一来,我们分析上两个问题:


 

        对于s1,先放入char类型,由于char类型是1个字节,小于8(VS默认值),所以直接放在0地址处。

        值得一提,长度是1的char无论何时都不会浪费空间,它会直接顶着上一个数据的存储空间放置。

        int型,大小4字节,小于8(VS默认值),所以放在sizeof(int)的整数倍的地址处,也就是4开始,到7结束。

        char 紧跟着int型;


        由于最大对齐数是4,此时结构体的最小大小大于8,要满足4的整数倍,那么这个结构体的大小向上取 是12。

        对于s2 :

        放入两个char,一个int后的结构体最小大小正好是8,那么结构体总大小就是8。

 

 


练习: 

int main(int argc, char* argv[])
{
  struct tagTest1
  {
    short a;
    char d; 
    long b;   
    long c;   
  };
  struct tagTest2
  {
    long b;   
    short c;
    char d;
    long a;   
  };
  struct tagTest3
  {
    short c;
    long b;
    char d;   
    long a;   
  };
  struct tagTest1 stT1;
  struct tagTest2 stT2;
  struct tagTest3 stT3;

  printf("%d %d %d", sizeof(stT1), sizeof(stT2), sizeof(stT3));
  return 0;

        结果: 


 完~

未经作者同意禁止转载

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/233870.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

108. 将有序数组转换为二叉搜索树

给你一个整数数组 nums &#xff0c;其中元素已经按 升序 排列&#xff0c;请你将其转换为一棵 高度平衡 二叉搜索树。 高度平衡 二叉树是一棵满足「每个节点的左右两个子树的高度差的绝对值不超过 1 」的二叉树。 示例 1&#xff1a; 输入&#xff1a;nums [-10,-3,0,5,9] 输…

06-React组件 Redux React-Redux

React组件化&#xff08;以Ant-Design为例&#xff09; 组件化编程&#xff0c;只需要去安装好对应的组件&#xff0c;然后通过各式各样的组件引入&#xff0c;实现快速开发 我们这里学习的是 Ant-design &#xff08;应该是这样&#xff09;&#xff0c;它有很多的组件供我们…

kali linux使用Proxmark3

其实kali linux下已经集成了Proxmark3命令&#xff0c;但是由于Proxmark3是开源设备&#xff0c;有时候系统默认安装的版本并不能很好的使用&#xff0c;因此需要手动编译最新的版本。 step 1 准备Proxmark3编译环境&#xff0c;因为kali linux比较激进&#xff0c;很多老旧的…

【EI会议征稿中】第三届信号处理与通信安全国际学术会议(ICSPCS 2024)

第三届信号处理与通信安全国际学术会议&#xff08;ICSPCS 2024&#xff09; 2024 3rd International Conference on Signal Processing and Communication Security 信号处理和通信安全是现代信息技术应用的重要领域&#xff0c;近年来这两个领域的研究相互交叉促进&#xf…

基于YOLOv7算法的高精度实时海上船只目标检测识别系统(PyTorch+Pyside6+YOLOv7)

摘要&#xff1a;基于YOLOv7算法的高精度实时海上船只目标检测系统可用于日常生活中检测与定位海上船只目标&#xff0c;此系统可完成对输入图片、视频、文件夹以及摄像头方式的目标检测与识别&#xff0c;同时本系统还支持检测结果可视化与导出。本系统采用YOLOv7目标检测算法…

配置BFD多跳检测示例

BFD简介 定义 双向转发检测BFD&#xff08;Bidirectional Forwarding Detection&#xff09;是一种全网统一的检测机制&#xff0c;用于快速检测、监控网络中链路或者IP路由的转发连通状况。 目的 为了减小设备故障对业务的影响&#xff0c;提高网络的可靠性&#xff0c;网…

【亲测有效】支持横竖屏 微信小程序video禁止进度条拖动,微信小程序遮罩进度条,

背景&#xff1a;部分课程禁止客户拖动视频进度条直至播放结束 红色是遮罩区域遮罩区域 实际遮罩效果&#xff08;有一个很浅的阴影区域&#xff09; 实现代码 .wxml文件 <video enable-progress-gesture"false" ><cover-view class"cover">…

JAVA全栈开发 day18MySql03

一、复习 为什么要用数据库数据库好处数据库的发展史​ 层次模型​ 网状模型​ 关系模型&#xff08;二维表专门存储数据&#xff0c; 表与表的关联&#xff09;​ 表与表的关系&#xff1a; 1对1 &#xff0c;1对多&#xff0c;多对多​ 非关系模型关系模…

Linux常见压缩指令小结

为什么需要压缩技术 我们都知道文件是以byte作为单位的&#xff0c;如果我们的文件仅仅在低位占一个1 0000 0001这种情况我们完全可以压缩一下&#xff0c;将高位的0全部抹掉即可。 如上所说是一种压缩技术&#xff0c;还有一种就是将1111(此处省略96个)一共100个1&#xff0…

Unity中Shader黑白阀值后处理效果

文章目录 前言一、我们先来PS看一下黑白阀值的效果二、使用step(a,b)函数实现效果三、实现脚本控制黑白阀值1、在Shader属性面板定义控制阀值变量2、把step的a改为_Value3、在后处理脚本设置公共成员变量,并且设置范围为&#xff08;0&#xff0c;1&#xff09;4、在Graphics.B…

vulnhub靶机Prime-2

下载地址&#xff1a;Prime (2021): 2 ~ VulnHub 主机发现 目标145 端口扫描 端口服务扫描 漏洞扫描 先去看一下80 扫目录 发现点不一样的 Wordpress&#xff08;记住还是要用api&#xff09; 好洞出来了看看利用方法 192.168.21.145/wp/wp-content/plugins/gracemedia-media…

【刷题篇】动态规划(六)

文章目录 1、最大子数组和2、环形子数组的最大和3、乘积最大子数组4、乘积为正数的最长子数组长度5、 等差数列划分6、最长湍流子数组 1、最大子数组和 给你一个整数数组 nums &#xff0c;请你找出一个具有最大和的连续子数组&#xff08;子数组最少包含一个元素&#xff09;&…

Proteus仿真--基于12864的自制硬件汉字库的应用

本文介绍基于12864的自制硬件汉字库的应用设计&#xff08;完整仿真源文件及代码见文末链接&#xff09; 其中12864LCD中显示的汉字是存储在2块AT24C1024芯片中 仿真图如下 仿真运行视频 Proteus仿真--基于12864的自制硬件字库的应用 附完整Proteus仿真资料代码资料 链接&am…

陀螺仪LSM6DSV16X与AI集成(4)----Qvar触摸电容配置

陀螺仪LSM6DSV16X与AI集成.4--Qvar触摸电容配置 概述视频教学样品申请源码下载生成STM32CUBEMX串口配置IIC配置CS和SA0设置串口重定向参考程序初始换管脚获取ID复位操作BDU设置Qvar 功能的实现和配置设置量程和速率配置过滤链激活 Qvar 功能获取Qvar数据演示 概述 Qvar&#x…

onnxruntime和tensorrt多batch推理

以lenet网络为例。 onnxruntime多batch推理 当batch size为2时&#xff0c;导出如下结构的onnx文件&#xff1a; python推理&#xff1a; import cv2 import numpy as np import onnxruntimeimg0 cv2.imread("2.png", 0) img1 cv2.imread("10.png", …

【MATLAB】基于EEMD分解的信号去噪算法(基础版)

代码操作 【MATLAB】基于EEMD分解的信号去噪算法&#xff08;基础版&#xff09; 代码的主要内容 基于EEMD&#xff08;集合经验模态分解&#xff09;的信号去噪算法通常可以结合相关系数、信号的熵值或者方差贡献率来完成去噪处理。这些指标可以用于确定阈值&#xff0c;从而…

Android:java.lang.RuntimeException: Unable to start activity ComponentInfo

java.lang.RuntimeException: Unable to start activity ComponentInfo 报错描述&#xff1a; 在导入别人项目运行时出现了这个报错&#xff1a; java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.news/com.example.activity.DetailNews}: ja…

SpringMVC修炼之旅(3)REST风格与拦截器

一、概述 1.1简介 Restful就是一个资源定位及资源操作的风格。不是标准也不是协议&#xff0c;只是一种风格。基于这个风格设计的软件可以更简洁&#xff0c;更有层次&#xff0c;更易于实现缓存等机制。 1.2功能 资源&#xff1a;互联网所有的事物都可以被抽象为资源 资源操作…

C++之获取变量信息名称、类型typeid

摘要 对于C工程量级比较庞大的代码&#xff0c;代码中的变量、类、函数、结构体的识别都是一件让人头疼的事情&#xff0c;一方面代码越写越多&#xff0c;内容越来越丰富&#xff0c;但是没有办法对已有的代码框架进行高度的整合提炼&#xff1b;另一方面对新人逐渐不友好&am…

C++笔记之通过静态类成员变量的方式在不同的类之间传递参数

C笔记之通过静态类成员变量的方式在不同的类之间传递参数 code review! 在C中&#xff0c;可以使用静态类成员变量作为一种在不同类之间传递参数的方式。静态类成员变量是类的所有对象之间共享的变量&#xff0c;它们存在于类的内部&#xff0c;但不属于任何特定的类对象。 …