结构体联合体枚举和位段

文章目录

  • 结构体
      • 结构体类型的声明
        • 特殊的声明
      • 结构的自引用
      • 结构体变量的定义和初始化
      • 结构体`内存对齐`
      • 为什么要内存对齐
      • 结构体传参
      • 结构体实现位段(位段的填充&可移植性)
        • 位段
        • 位段的内存分配
        • 空间如何开辟
        • 位段的跨平台问题
        • 位段的应用
  • 枚举
      • 枚举类型的定义
      • 枚举的`优点`
      • 枚举的使用
  • 联合体
      • 联合类型的定义
      • 联合的特点
      • 联合大小的计算

c中有内置类型,也有自定义类型
自定义类型有:结构体 联合体 枚举
结构是一些值的集合,这些值称为成员变量。结构的每个成员可以是不同类型的变量。
就像之前学过的数组一样:一组相同类型元素的集合

结构体

结构体类型的声明

struct是关键字
比如描述一个学生的信息
只是创建了结构体类型
分号不能丢

struct Stu//Stu结构体名字
{
char name[20];//结构体成员}s1,s2;//s1,s2是全局变量

全局还是局部取决于是在主函数之内还是之外

特殊的声明

在声明结构的时候,可以不完全的声明。
匿名结构体类型

struct
{
char name[20];//结构体成员
}s1,s2

只能使用一次

//匿名结构体类型
struct
{int a;char b;float c;
}x;
struct
{int a;char b;float c;
}a[20], *p;
//在上面代码的基础上,下面的代码合法吗?
p = &x

会报警告,因为会认为两个结构体空间不一样
非法

结构的自引用

结构体不能包含同结构体类型
能包括同结构体类型指针
比如设置链表的节点

struct Node
{
int data;
struct Node* next;
};

链表就是用指针像链条一样把数据穿起来
数据结构会学

//代码3
typedef struct
{int data;
struct Node* next;
}Node;
//这样写代码,可行否?

typedef需要结构体存在才能重新定义
而Node还没创建,就已经用上了
先有鸡还是先有蛋的问题~

//`解决方案`:
typedef struct Node
{int data;struct Node* next;
}Node;

这时候Node=struct Node

结构体变量的定义和初始化

像上面的s1 s2 Node就是定义的变量
声明类型的同时定义变量
struct Stu p2; //定义结构体变量p2
初始化

struct Point
{int x;int y;
}p1; 声明类型的同时定义变量p1
struct Point p2; 定义结构体变量p2
初始化:定义变量的同时赋初值。
struct Point p3 = {x, y};
struct Stu        类型声明
{char name[15];名字int age;      年龄
};
struct Stu s = {"zhangsan", 20};初始化
struct Node
{int data;struct Point p;struct Node* next; 
}n1 = {10, {4,5}, NULL}; 结构体嵌套初始化
struct Node n2 = {20, {5, 6}, NULL};结构体嵌套初始化

打印

.操作符直接访问
->间接访问

#define  _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdio.h>
#include<string.h>
#if 0
struct Stu
{//结构体成员char name[18];int age;double score;};
void set_stu(struct Stu* ps)//结构体指针变量
{strcpy(ps->name, "chenyanan");//ps->name和(*ps).name是一样的ps->age = 18;ps->score = 100.0;
}
void print_stu(struct Stu s)
{printf("%s %d %lf\n", s.name, s.age, s.score);//.操作符 结构对象.成员
}
int main()
{struct Stu s = { 0 };//类型+变量名即s是结构体的对象set_stu(&s);//形参和实参的形式不同,如果(s),实参未改变,应传递地址print_stu(s);return 0;
}

结构体内存对齐

在这里插入图片描述
在这里插入图片描述

计算偏移量
在这里插入图片描述
头文件include<stddef.h>

在这里插入图片描述

struct S3
{double d;char c;int i;
};struct S2
{char c1;struct S3 s3;double d;
};
int main()
{printf("%d\n", offsetof(struct S2, c1));//0printf("%d\n", offsetof(struct S2, s3));//8printf("%d\n", offsetof(struct S2, d));//24printf("%d\n", sizeof(struct S2));//32
}

如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整
体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。
最大对齐数是8

为什么要内存对齐

  1. 平台原因(移植原因)
    不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。
    2. 性能原因
    数据结构(尤其是栈)应该尽可能地在自然边界上对齐。
    原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访
    问。
    总体来说:
    结构体的内存对齐是拿空间来换取时间的做法。
    提高效率
    静态区栈区也是同样的储存规则

那在设计结构体的时候,我们既要满足对齐,又要节省空间
如何做到: 让占用空间小的成员尽量集中在一起

默认最大对齐数可以修改
#pragma pack(对齐数)
但不要乱改

结构体传参

struct S
{int data[1000];int num;
};
struct S s = {{1,2,3,4}, 1000};
//结构体传参
void print1(struct S s)
{printf("%d\n", s.num);
}
//结构体地址传参
void print2(struct S* ps)
{printf("%d\n", ps->num);
}
int main()
{print1(s);  //传结构体print2(&s); //传地址return 0;
}

首选print2
传值调用会压栈,开辟一个大空间接收数据
效率低,性能下降
传址空间小,地址4/8个字节
传结构体的地址

结构体实现位段(位段的填充&可移植性)

位段

位段的声明和结构是类似的,有两个不同:
1.位段的成员必须是 int、unsigned int 或signed int 。
2.位段的成员名后边有一个冒号和一个数字。

struct A
{int _a:2;//_a分配2个biteint _b:5;int _c:10;int _d:30;
};

位段用来节省空间

位段的内存分配
  1. 位段的成员可以是 int unsigned int signed int 或者是 char (属于整形家族)类型
  2. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的。
    3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段
struct A
{int _a:2;//_a分配2个byteint _b:5;int _c:10;//15//开辟四个字节32个byteint _d:30;
};

47byte
int 4个字节 32byte
不够用再拿四个字节
会用完剩下的15个byte还是用新开辟的byte?

一个例子


struct S
{char a:3;char 8个byte剩下5char b:4;1char c:5;开辟一个字节,是会把1byte用掉在用新开辟的byte呢?还是会浪费掉呢?若浪费打印的是3个字节,没浪费打印2个字节char d:4;
};int main()
{
printf("%d\n",sizeof(struct S)):
return 0;
}

在这里插入图片描述

不够字节后,接下来用的是新开辟的空间

空间如何开辟

大小端是超过一个字节,内存存放顺序的问题
在这里插入图片描述
这是在VS2022中的

位段的跨平台问题

在这里插入图片描述
总结

跟结构体相比,位段能达到同种效果,且能节省空间,但有跨平台问题,解决就是要能针对不同平台写不同的代码

位段的应用

在网络里传输数据的时候运用
在这里插入图片描述
基于网络协议写出来的网络编程是可以通信的
ip协议

枚举

枚举类型的定义

通过关键字enum来定义
基本语法

enum 枚举名{
成员1=1;
成员2=2;
...
成员n=值n;
};

枚举成员可以是任意标识符
值必须是整数
作用域在定义的文件内
若想要在其他文件内使用需要包含枚举的头文件
没有指定值,那默认第一个枚举成员的值为0
这里的enum没有存储到内存中,相当于是模板图纸,使用了才会有内存的空间分配

枚举的优点

我们可以使用 #define 定义常量,为什么非要使用枚举? 枚举的优点:

  1. 增加代码的可读性和可维护性
  2. 和#define定义的标识符比较枚举有类型检查,更加严谨
    宏在预处理阶段就把值进行了替换,在可执行文件中看到的也是替换之后的值,不便于观察
  3. 便于调试
  4. 使用方便,一次可以定义多个常量

枚举的使用

enum Color
{
red=1;
green=3;
orange=6;
};
int main()
{
//使用枚举类型变量
enum Color color;
//只能拿枚举常量给枚举变量赋值,才不会出现类型的差异。
color=red;
//不能color=1;
}

常用在需要设置特定顺序的常量值时 ,比如星期,月份,方向

联合体

联合类型的定义

包含一系列的成员,共用同一块内存空间->共用体

联合的特点

地址相同

#define  _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
union Un
{char a;int i;}u;
int main()
{printf("%zd\n", sizeof(u));printf("%p\n", &u);printf("%p\n", &(u.i));printf("%p\n", &(u.a));return 0;
}

在这里插入图片描述

在同一块内存中存储不同类型
不能同时用,改变一个也会改变另一个

联合大小的计算

联合的大小至少是最大成员的大小。
当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍

union Un1
{char c[5];int i;
};
union Un2
{short c[7];int i;
};
//下面输出的结果是什么?
printf("%d\n", sizeof(union Un1));8
printf("%d\n", sizeof(union Un2));16

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

Pyqt5中,QGroupBox组件标题字样(标题和内容样式分开设置)相对于解除继承

Python代码示例&#xff1a; import sys from PyQt5.QtWidgets import QApplication, QWidget, QVBoxLayout, QGroupBox, QLabelclass MyApp(QWidget):def __init__(self):super().__init__()# 创建一个 QVBoxLayout 实例layout QVBoxLayout()# 创建 QGroupBox 实例self.grou…

微信小程序原生<map>地图实现标记多个位置以及map 组件 callout 自定义气泡

老规矩先上效果图: 1 、在pages文件夹下新建image文件夹用来存放标记的图片。 2、代码片段 也可以参考小程序文档:https://developers.weixin.qq.com/miniprogram/dev/component/map.html index.wxml代码 <mapid="map"style="width: 100%; height:100%;&…

ARM汇编与逆向工程:蓝狐卷基础知识

与传统的CISC&#xff08;Complex Instruction Set Computer&#xff0c;复杂指令集计算机&#xff09;架构相比&#xff0c;Arm架构的指令集更加简洁明了&#xff0c;指令执行效率更高&#xff0c;能够在更低的功耗下完成同样的计算任务&#xff0c;因此在低功耗、嵌入式等领域…

[嵌入式系统-39]:龙芯1B 开发学习套件 -10-PMON启动过程Sstart.S详解

目录 一、龙芯向量表与启动程序的入口&#xff08;复位向量&#xff09; 1.1 复位向量&#xff1a; 1.2 代码执行流程 1.3 计算机的南桥 VS 北桥 二、PMON代码执行流程 三、Start.S详解 3.1 CPU初始化时所需要的宏定义 &#xff08;1&#xff09;与CPU相关的一些宏定义…

2024人工智能四大趋势→

2023年&#xff0c;世人见证了ChatGPT在全球范围的大火。以生成式人工智能为代表的新一代人工智能问世&#xff0c;改变了人工智能&#xff08;AI&#xff09;技术与应用的发展轨迹&#xff0c;加速了人与AI的互动进程&#xff0c;是人工智能发展史上的新里程碑。2024年&#x…

皂液器问卷调查

媳妇非要买这种皂液器&#xff0c;来问问友友们有用过的帮忙识别一下是否是真的好用&#xff1a;皂液器问卷调查 4个题

SMART PLC 卷径计算(圈数检测+膜厚叠加法)

1、卷径计算(膜厚叠加+数值积分器应用博途PLC SCL代码) https://rxxw-control.blog.csdn.net/article/details/136719982https://rxxw-control.blog.csdn.net/article/details/1367199822、膜厚叠加法 https://rxxw-control.blog.csdn.net/article/details/128600466

[题解]无厘头题目——无聊的军官

这道题非常无厘头&#xff01; 题目描述&#xff1a; 每个学年的开始&#xff0c;高一新生们都要进行传统的军训。今年有一个军训教官十分奇怪&#xff0c;他为了测试学员们的反应能力&#xff0c;每次吹哨后学员们都会变换位置。每次左数第I位学员都会站到第ai个位置&#x…

60种常用可视化图表的使用场景——(上)

文章目录 1、点阵图 2、点数图 3、弧线图 4、折线图 5、平行坐标图 6、网络图 7、象形图 8、直方图 9、密度图 10、人口金字塔 11、条形图 12、多组条形图 13、堆叠式条形图 14、不等宽柱状图 15、面积图 16、比例面积图 17、堆叠式面积图 18、量化波形图 19、雷达图 20、桑基图…

P8665 [蓝桥杯 2018 省 A] 航班时间:格式问题

题目链接&#xff1a; P8665 [蓝桥杯 2018 省 A] 航班时间 哎&#xff0c;这道题做了很长时间还是做错了&#x1f940; 代码 #include<iostream> #include<algorithm> using namespace std; int get() {int h1,m1,s1,h2,m2,s2,day0;scanf("%d:%d:%d %d:%d:…

重学SpringBoot3-整合SSM

更多SpringBoot3内容请关注我的专栏&#xff1a;《SpringBoot3》 期待您的点赞&#x1f44d;收藏⭐评论✍ 重学SpringBoot3-整合SSM Spring Boot整合SSM示例1. 创建Spring Boot项目2. 配置数据源3. 配置MyBatis4. 实现数据访问对象&#xff08;DAO&#xff09;5. 编写服务层和控…

Tiktok在线网页版和Tiktok安卓解锁版教程(Tiktok免登录国内直接看)

TikTok和抖音作为众所周知的一对兄弟&#xff0c;所开创的市场前景不可估量。它们不仅颠覆了很多传统认知&#xff0c;也直接让更多人接受了这些新奇事物。然而&#xff0c;TikTok的版本受限于国外&#xff0c;需要特定网络环境&#xff0c;一旦识别为国内&#xff0c;将无法使…

18.古今成大事者,必以多选替身为第一要义——代理模式详解

“杏市而外&#xff0c;尚有何人可以分统?亦须早早提拔。办大事者以多多选替手为第一义&#xff0c;满意之选不可得&#xff0c;姑节取其次&#xff0c;以待徐徐教育可也。 ——曾国藩同治元年四月十二日” 一言 代理模式核心思想是为对象提供一个替身&#xff0c;以控制对这…

JDK8和JDK11在Ubuntu18上切换(解决nvvp启动报错)

本文主要介绍JDK8和JDK11在Ubuntu18上切换&#xff0c;以供读者能够理解该技术的定义、原理、应用。 &#x1f3ac;个人简介&#xff1a;一个全栈工程师的升级之路&#xff01; &#x1f4cb;个人专栏&#xff1a;计算机杂记 &#x1f380;CSDN主页 发狂的小花 &#x1f304;人…

王道机试C++第8章递归与分治 Day35和蓝桥杯两道真题程序

第 8 章 递归与分治 递归是指&#xff1a;函数直接或间接调用自身的一种方法&#xff0c;通常可把一个复杂的大型问题层层转化为与原问题相似但规模较小的问题来求解。 递归策略只需少量的程序就可描述解题过程所需的多次重复计算&#xff0c;因此大大减少了程序的代码量。 8.…

Nacos安装与集群搭建

Nacos安装与集群搭建 Nacos安装指南1.Windows安装1.1.下载安装包1.2.解压1.3.端口配置1.4.启动1.5.访问 2.Linux安装2.1.安装JDK2.2.上传安装包2.3.解压2.4.端口配置2.5.启动 3.Nacos的依赖Nacos集群搭建1.集群结构图2.搭建集群2.1.初始化数据库2.2.配置Nacos2.3.启动2.4.nginx…

ts的interface和type区别

1. 场景 interface 是用来描述对象类型的结构&#xff0c;可以定义对象的属性名和属性值的类型&#xff0c;也可以定义函数类型。interface User {name: string;age: number;sayHello(): void; } const user: User {name: "",age: 2,sayHello() {...} }可以用这个U…

mysql 更新时,旧值与新值相同会怎么做?

文章目录 1 问题描述2 验证2.1 验证猜想12.2 验证猜想2 3 结论4 mysql 为什么这么设计呢&#xff1f; 1 问题描述 创建一张表t&#xff0c;插入一行数据 mysql> CREATE TABLE t ( id int(11) NOT NULL primary key auto_increment, a int(11) DEFAULT NULL ) ENGINEInnoDB…

Nexus 部署使用

1. 简介 Nexus是Maven仓库管理器&#xff0c;也可以叫Maven的私服&#xff0c;是私服的一种。Nexus是一个强大的Maven仓库管理器&#xff0c;它极大地简化了自己内部仓库的维护和外部仓库的访问。利用Nexus你可以只在一个地方就能够完全控制访问和部署在你所维护仓库中的每个A…

Sonarqube中Java规则与CWE与OWASP的映射关系

很多企业使用Sonarqube社区版作为静态分析工具&#xff0c;在开发阶段检测代码中的缺陷或安全漏洞。但是如果是作为SAST工具厂商&#xff0c;集成该引擎&#xff0c;则需要把Sonarqube中的检测规则与其它引擎的规则进行整合&#xff0c;例如下图&#xff0c;把Sonarqube中的一些…