自定义类型:结构体
自定义类型:结构体
- 一:引入
- 二:结构体类型的声明
- 1:正常声明
- 2:特殊声明
- 三:结构体变量的创建和初始化
- 1:结构体变量的创建
- 2:结构体变量的初始化
- 三:结构体访问操作符
- 四:结构体内存对齐
- 1:对齐规则
- 2:为什么存在内存对齐?
- 3:修改默认对齐数
- 五:结构体传参
- 六:结构体实现位段
- 1:位段的声明:
- 2:位段的内存分配:
- 3:位段的跨平台问题
- 4:位段的应用
- 5:位段的使用注意事项:
一:引入
C语言中有内置类型,如int ,char,float,long,long long等,这些内置类型可以直接使用。
如果我们想要描述一个学生,比如学生的名字,年龄,分数等,只有一个内置类型是无法描述清楚的,所以引入结构体的使用。
二:结构体类型的声明
1:正常声明
结构体是自定义类型
结构体是一些值的集合,这些值称为成员变量,结构体的每个成员可以是不同类型的变量。
例如描述一个同学:
struct Stu//
{char name[20];//名字int age;//年龄float score;//分数};
{}里面的内容就是结构体的成员。
2:特殊声明
在声明结构体的时候可以不完全声明。
struct
{int a;char b;float c;
}s = { 0 };//这种特殊的声明只能此时初始化
struct
{int a;char b;float c;
}*ps;
int main()
{ps = &s;//error,虽然结构类型的成员一模一样,但编译器依然认为=两边是不同的指针类型return 0;
}
三:结构体变量的创建和初始化
1:结构体变量的创建
#include <stdio.h>
struct stu
{char name[20];int age;float score;
}s4, s5;//结构体变量的创建s4,s5
int main()
{struct stu s1, s2, s3; //结构体变量的创建s1,s2,s3return 0;
}
2:结构体变量的初始化
#include <stdio.h>
struct stu
{char name[20];int age;float score;
}s2 = { "lisi",21,89.5f };//结构体变量的创建及初始化
int main()
{struct stu s1 = { "zhangsan",20,85.5f };//结构体变量的创建及初始化struct stu s2={.age=22,.name="wanger",.score=98.5f};//特殊初始化return 0;}
struct Stu//struct是结构体类型的关键字//struct Stu是用户定义的结构体类型
{//name,age,score都是都是结构体成员名char name[20];//名字int age;//年龄int score;//分数}s;//声明的同时定义结构体变量s
//结构体类型不占用空间,相当于盖房子时所使用的图纸
struct Stu s = { "zhangsan",20,84 };//结构体变量的初始化
//结构体变量占用空间,相当于盖好了的房子
三:结构体访问操作符
结构体成员访问操作符有两个,一个是 . 一个是->
形式如下:
结构体变量.成员变量名
结构体指针->成员变量名
#include <stdio.h>
struct stu
{char name[20];int age;int score;
};
int main()
{struct stu s = { "zhangsan",20,90 };struct stu* p = &s;printf("%d %d\n", s.age,s.score);printf("%d %d\n", p->age, p->score);return 0;
}
四:结构体内存对齐
结构体的大小与结构体内存对齐有关。
1:对齐规则
1:结构体的第一个成员对齐到相对结构体变量起始位置偏移量为0的地址处。
2:其他成员变量要对齐到对齐数的整数倍的地址处。
对齐数=编译器默认的对齐数(VS默认的是8)与该成员变量大小的较小值.
3:结构体总大小为最大对齐数(结构体中的每个成员都有一个对齐数,所有对齐数中最大的)的整数倍。
4:如果嵌套了结构体的情况,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体成员的对齐数)的整数倍。
例1:
#include <stdio.h>
struct S1
{char c1;char c2;int a;
};struct S2
{char c1;int a;char c2;
};int main()
{struct S2 s2 = { 'a', 100, 'b'};printf("%zd\n", sizeof(s2));struct S1 s1 = { 'a', 'b', 100 };printf("%zd\n", sizeof(s1));return 0;
}


例2:结构体的嵌套
#include <stdio.h>
struct S3
{double d;char c;int i;
};struct S4
{char c1;struct S3 s3;double d;
};int main()
{printf("%zd\n", sizeof(struct S3));//?printf("%zd\n", sizeof(struct S4));//?return 0;
}


2:为什么存在内存对齐?
1:平台原因:不是所有的硬件平台都能访问任意地址上的任意数据的,某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。
2:性能原因:数据结构(尤其是栈)应该尽可能地在自然边界上对齐。原因在于为了访问未对齐的内存,处理器需要两次内存访问;而对齐的内存访问仅需要一次访问。假设一个处理器总是从内存中取8个字节,则地址必须是8的倍数。如果我们能保证所有的double类型的数据的地址都对齐成8的倍数,那么就可以用一个内存操作数 来读或者写值了。否则我们可能执行两次内存访问,因为对象可能被分放在两个8字节内存块中。
总体来说:结构体的内存对齐是拿空间来换取时间的做法。
在设计结构体的时候,我们既要满足对齐,又要节省空间,可以让内存占用少的成员尽量放在一起。
#include <stdio.h>
struct S1
{char c1;char c2;int a;
};struct S2
{char c1;int a;char c2;
};int main()
{struct S2 s2 = { 'a', 100, 'b'};printf("%zd\n", sizeof(s2));//12struct S1 s1 = { 'a', 'b', 100 };printf("%zd\n", sizeof(s1));//8两个char类型的数据放在了一起return 0;
}
3:修改默认对齐数
#pragma这个预处理命令可以改变编译器的默认对齐数。
#pragma pack()//取消设置的默认对齐数,还原为系统默认的对齐数
#include <stdio.h>
#pragma pack(1)//将默认对齐数设置为1
struct s
{char c1;int i;char c2;
};int main()
{printf("%d ", sizeof(struct s));return 0;
}
五:结构体传参
#include <stdio.h>
struct S
{int age;int num;
};
struct S s = { 20,100 };
void print1(struct S t)
{printf("%d\n", t.num);
}
void print2(struct S* ps)
{printf("%d\n", ps->num);
}
int main()
{print1(s);//传递的是结构体变量,也就是传值调用print2(&s);//传递的是结构体变量的地址,也就是传址调用return 0;
}
传递结构体变量的时候,形参是实参的一份临时拷贝,形参会单独占用空间,传数据
也需要时间,如果实参数据较大,那么浪费的空间和时间将非常大。
函数传参的时候,参数需要压栈,会有时间和空间上的系统开销,如果传递一个结构体对象的时候,结构体过大,参数压栈的系统开销比较大,会导致性能的下降。
六:结构体实现位段
位段是基于结构体的,结构体可能浪费空间,而位段可以节省空间。位段的位指的是二进制位。
1:位段的声明:
#include <stdio.h>
struct A
{int a : 2;//a占2个bit位int b : 4;//b占4个bit位int c : 7;//c占7个bit位
};
int main()
{printf("%d ", sizeof(struct A));return 0;
}
2:位段的内存分配:
1:位段的成员可以是int ,unsigned int,signed int,char类型。
2:位段的空间上是按照需要4个字节(int)或一个字节(char)来开辟的。
3:位段涉及很多不确定的因素,位段是不可以跨平台的,如果想要跨平台需要使用不同的代码。(当开辟了内存后,内存中的每个bit从左向右使用还是从右向左使用,不确定;当前面的内存使用后,剩下的空间不能够放下一个成员时,剩下的空间是否使用,也不确定)。
下面主要讲述在VS编译器下位段的内存分配。
在VS编译器下,每个bit位从右向左使用,使用后剩余的空间不能够放下下一个元素时,该剩余空间将不再使用,需要再开辟一个字节的空间,放下一个元素。
#include <stdio.h>
struct S
{char a : 3;//开辟第一个字节,a占3个bit位,还剩5个比特位char b : 4;//剩下的5个bit位能够放下b,还剩下1个bit位char c : 5;//剩下的1个比特位 不能够放下c,再开辟一个新的字节放c,//放c后还剩下3个字节,剩下的3个字节放不下dchar d : 4;//再开辟一个新的字节放d
};int main()
{struct S s = { 0 };printf("%d ", sizeof(s));//3return 0;
}
3:位段的跨平台问题
1:int位段被当成有符号数还是无符号数是不确定的。
2:位段中最大的数目不能确定(16位机器下最大是16,32位机器下是32,写成33,在16位和32位的机器下都会报错)

3:位段中的成员在内存中是从左向右分配还是从右向左分配不确定。
4:当一个结构包含2个位段,第二个位段的成员比较大,第一个位段剩下的空间无法容纳第二个位段时,是舍弃剩余的位还是使用,这是不确定的。
总之:位段可以节省空间,但有跨平台问题,想要解决跨平台问题,需要写不同的代码
4:位段的应用
在网络协议中,IP数据报的格式,其中很多的属性只需几个bit位就能够描述,在这里使用位段,可以节省很多空间,网络传输的数据报大小也会较小一些,对网络的畅通是有帮助的。

5:位段的使用注意事项:
内存中一个字节有一个编号(地址),位段可能几个成员共用一个字节,这样有些成员的起始位置并不是某个字节的起始位置,那么这些位置是没有地址的。所以不能对位段的成员使用&操作符,这样就不能使用scanf直接给位段的成员输入值,只能先输入到一个变量里,然后赋值给位段的成员。

#include <stdio.h>
struct S
{char a : 3;char b : 5;int c : 7;
};
int main()
{struct S s = { 0 };int c = 0;//scanf("%d", &s.c);scanf("%d", &c);s.c = c;printf("c=%d ", s.c);return 0;
}
相关文章:
自定义类型:结构体
自定义类型:结构体 一:引入二:结构体类型的声明1:正常声明2:特殊声明 三:结构体变量的创建和初始化1:结构体变量的创建2:结构体变量的初始化 三:结构体访问操作符四:结构…...
postman如何设置才能SwitchHosts切换host无缓存请求到指定ip服务
开发测试中,遇到多版本同域名的服务使用postman进行测试,一般会搭配SwitchHosts切换host类似工具进行请求,postman缓存比较重,如何做到无缓存请求呢,下面简单记录一下如何实现 首先要知道如何当前请求服务的ip是哪个 打开postman 依次点击/menu/view/show postman console 就…...
LeetCode LCR 103. 零钱兑换【完全背包,恰好装满背包的最小问题】中等
本文属于「征服LeetCode」系列文章之一,这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁,本系列将至少持续到刷完所有无锁题之日为止;由于LeetCode还在不断地创建新题,本系列的终止日期可能是永远。在这一系列刷题文章…...
竞赛 基于深度学习的人脸专注度检测计算系统 - opencv python cnn
文章目录 1 前言2 相关技术2.1CNN简介2.2 人脸识别算法2.3专注检测原理2.4 OpenCV 3 功能介绍3.1人脸录入功能3.2 人脸识别3.3 人脸专注度检测3.4 识别记录 4 最后 1 前言 🔥 优质竞赛项目系列,今天要分享的是 🚩 基于深度学习的人脸专注度…...
supervisord 进程管理器 Laravel执行队列
supervisord 进程管理器 执行队列 安装 yum install supervisor修改配置文件 /etc/supervisord.conf 最后一行 ini改为conf files=/etc/supervisor.d/*.conf vim /etc/supervisord.conf/etc/supervisord.d目录下新增配置文件 vim laravel-worker.conf 修改i 粘贴内容 退出修…...
Lnmp架构之mysql数据库实战1
1、mysql数据库编译 编译成功 2、mysql数据库初始化 配置数据目录 全局文件修改内容 生成初始化密码并进行初始化设定 3、mysql主从复制 什么是mysql的主从复制? MySQL的主从复制是一种常见的数据库复制技术,用于将一个数据库服务器(称为主…...
ChatGLM 大模型炼丹手册-理论篇
序言一)大还丹的崛起 在修真界,人们一直渴望拥有一种神奇的「万能型丹药」,可包治百病。 但遗憾的是,在很长的一段时间里,炼丹师们只能对症炼药。每一枚丹药,都是特效药,专治一种病。这样就导致,每遇到一个新的问题,都需要针对性的炼制,炼丹师们苦不堪言,修真者们吐…...
Spring Boot集成Redis实现数据缓存
🌿欢迎来到衍生星球的CSDN博文🌿 🍁本文主要学习Spring Boot集成Redis实现数据缓存 🍁 🌱我是衍生星球,一个从事集成开发的打工人🌱 ⭐️喜欢的朋友可以关注一下🫰🫰&…...
CentOS 7 安装Libevent
CentOS 7 安装Libevent 1.下载安装包 新版本是libevent-2.1.12-stable.tar.gz。(如果你的系统已经安装了libevent,可以不用安装) 官网:http://www.monkey.org/~provos/libevent/ 2.创建目录 # mkdir libevent-stable 3.解压 …...
线性代数的本质——几何角度理解
B站网课来自 3Blue1Brown的翻译版,看完醍醐灌顶,强烈推荐: 线性代数的本质 本课程从几何的角度翻译了线代中各种核心的概念及性质,对做题和练习效果有实质性的提高,下面博主来总结一下自己的理解 1.向量的本质 在物…...
SSH key 运作方式
1、本地创建SSH key pairs 2、把public key上传到网站服务器(如GitHub 3、当使用ssh方式连接时 本地SSH client向远端请求ssh连接远端发来random data要求加密本地ssh client用private key加密,把加密的data发送过去(不发送private key远端接…...
【基于MBD开发模式的matlab持续集成(一)】
基于MBD开发模式的matlab持续集成 引言 或许是感受到行业内卷的愈加激烈,在传统制造和高新技术相结合的新能源领域对软件工程开发的要求也愈加提高,尤其在互联网已经大行 其道的敏捷开发,便顺其自然的被新能源的老板们所看重。 概述 本文…...
Linux学习记录——이십팔 网络基础(1)
文章目录 1、了解2、网络协议栈3、TCP/IP模型4、网络传输1、同一局域网(子网)2、局域网通信原理3、跨一个路由器的两个子网4、其它 详细的网络发展历史就不写了 1、了解 为什么会出现网络?一开始多个计算机之间想要共享文件,就得…...
CSS动效合集之实现气泡发散动画
前言 👏CSS动效合集之实现气泡发散动画,速速来Get吧~ 🥇文末分享源代码。记得点赞关注收藏! 1.实现效果 2.实现步骤 定义一个数组bubbles,用来存储气泡列表的基本新,w表示宽高,x表示绝对定位…...
六、串口通信
六、串口通信 串口接口介绍使用串口向电脑发送数据电脑发送数据控制LED灯 串口接口介绍 SBUF是串口数据缓存器,物理上是两个独立的寄存器,但占用相同的地址。写操作时,写入的是发送寄存器;读操作时,读出的是接收寄存器…...
如何将 JavaScript Excel XLSX 查看器添加到Web应用程序
在 JavaScript 中创建 Excel 查看器可能是一项艰巨的任务,但使用 SpreadJS JavaScript 电子表格,创建过程要简单得多。在本教程博客中,我们将向您展示如何使用 SpreadJS 的强大功能来创建一个查看器,该查看器允许您在 Web 浏览器中…...
网安周报|CISA发布增强开源安全性的计划
1、CISA发布增强开源安全性的计划 美国一家领先的安全机构发布了一项期待已久的计划,详细说明了它将如何增强联邦政府和整个生态系统的开源安全性。美国网络安全和基础设施安全局(CISA)开源软件安全路线图在安全开源峰会上发布。据估计&#…...
使用 Docker 安装 Elasticsearch (本地环境 M1 Mac)
Elasticsearchkibana下载安装 docker pull elasticsearch:7.16.2docker run --name es -d -e ES_JAVA_OPTS“-Xms512m -Xmx512m” -e “discovery.typesingle-node” -p 9200:9200 -p 9300:9300 elasticsearch:7.16.2docker pull kibana:7.16.2docker run --name kibana -e EL…...
Visual Studio中MD与MT的区别及运行库类型选择
MT与MD的区别 /MT:是multithread-static version,是多线程静态版本的意思,项目会使用运行时库的多线程静态版本,编译器会将LIBCMT.lib放入.obj文件中,以便链接器使用LIBCMT.lib解析外部符号;/MTdÿ…...
Vue3函数式编程
文章目录 前言一、三种编程风格1.template2.jsx/tsx3.函数式编写风格 二、函数式编程1.使用场景2.参数3.例子3.render渲染函数 总结 前言 本文主要记录vue3中的函数式编程以及其他编程风格的简介 一、三种编程风格 1.template Vue 使用一种基于 HTML 的模板语法,…...
网络编程(Modbus进阶)
思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…...
大数据学习栈记——Neo4j的安装与使用
本文介绍图数据库Neofj的安装与使用,操作系统:Ubuntu24.04,Neofj版本:2025.04.0。 Apt安装 Neofj可以进行官网安装:Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...
Ubuntu系统下交叉编译openssl
一、参考资料 OpenSSL&&libcurl库的交叉编译 - hesetone - 博客园 二、准备工作 1. 编译环境 宿主机:Ubuntu 20.04.6 LTSHost:ARM32位交叉编译器:arm-linux-gnueabihf-gcc-11.1.0 2. 设置交叉编译工具链 在交叉编译之前&#x…...
循环冗余码校验CRC码 算法步骤+详细实例计算
通信过程:(白话解释) 我们将原始待发送的消息称为 M M M,依据发送接收消息双方约定的生成多项式 G ( x ) G(x) G(x)(意思就是 G ( x ) G(x) G(x) 是已知的)࿰…...
【JVM】- 内存结构
引言 JVM:Java Virtual Machine 定义:Java虚拟机,Java二进制字节码的运行环境好处: 一次编写,到处运行自动内存管理,垃圾回收的功能数组下标越界检查(会抛异常,不会覆盖到其他代码…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
Mobile ALOHA全身模仿学习
一、题目 Mobile ALOHA:通过低成本全身远程操作学习双手移动操作 传统模仿学习(Imitation Learning)缺点:聚焦与桌面操作,缺乏通用任务所需的移动性和灵活性 本论文优点:(1)在ALOHA…...
jmeter聚合报告中参数详解
sample、average、min、max、90%line、95%line,99%line、Error错误率、吞吐量Thoughput、KB/sec每秒传输的数据量 sample(样本数) 表示测试中发送的请求数量,即测试执行了多少次请求。 单位,以个或者次数表示。 示例:…...
水泥厂自动化升级利器:Devicenet转Modbus rtu协议转换网关
在水泥厂的生产流程中,工业自动化网关起着至关重要的作用,尤其是JH-DVN-RTU疆鸿智能Devicenet转Modbus rtu协议转换网关,为水泥厂实现高效生产与精准控制提供了有力支持。 水泥厂设备众多,其中不少设备采用Devicenet协议。Devicen…...
Matlab实现任意伪彩色图像可视化显示
Matlab实现任意伪彩色图像可视化显示 1、灰度原始图像2、RGB彩色原始图像 在科研研究中,如何展示好看的实验结果图像非常重要!!! 1、灰度原始图像 灰度图像每个像素点只有一个数值,代表该点的亮度(或…...
