C语言:结构体

目录

结构体类型的声明

匿名结构体

全局结构体变量

嵌套结构体

访问结构体成员

结构的自引用

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

结构体内存对齐

结构体内存对齐规则

修改默认对齐数 #pragma pack(n) 

offsetof 求结构体成员相对于结构体开头的偏移量的宏。 

为什么存在内存对齐?

结构体传参

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

什么是位段

位段的内存分配规则

位段的跨平台问题

位段的应用


结构体类型的声明

匿名结构体

/* 匿名结构体类型——只能用一次,即在声明的时候顺带创建一个变量 */
struct
{char name[20];int age;
} s4;

全局结构体变量

struct Stu
{char name[20];int age;
} s1, s2; // s1 s2是全局变量struct Stu
{char name[20];int age;
} s1 = {"zhanghai", 22}; // 声明并初始化全局变量s1

嵌套结构体

struct Score
{int score;char str[20];
};struct Stu
{char name[20];int age;struct Score s;
};int main()
{// 初始化局部变量s3struct Stu s3 = {"zhanghai", 22, {100, "q"}};printf("%s %d %d %s", s3.name, s3.age, s3.s.score, s3.s.str); // zhanghai 22 100 qreturn 0;
}

访问结构体成员

#include <stdio.h>/* 本例论如何声明结构体,并使用三种方式访问结构体各成员 */struct MyType
{/* 成员 */char name[20];int age;char sex[10];char tele[12];
};/* 接受结构体指针地址,struct MyType 类型int,表示一种类型;  *p是因为要指针赋值给指针变量时需加* */
void myPrintf(struct MyType *p)
{/* *p 表示使用*将指针变量p解引用,得到 *p == obj(可参考09_pointer) */printf("%s %d %s %s\n", (*p).name, (*p).age, (*p).sex, (*p).tele); // "zhanghai", 20, "nan", "15556201597"/* p 表示指针变量,结构体指针变量 -> 成员名,也就是结构体指针变量若想访问成员,需要操作符 -> */printf("%s %d %s %s\n", p->name, p->age, p->sex, p->tele); // "zhanghai", 20, "nan", "15556201597"
}int main()
{struct MyType obj = {"zhanghai", 20, "nan", "15556201597"};// 结构体对象.成员名printf("%s %d %s %s\n", obj.name, obj.age, obj.sex, obj.tele); // "zhanghai", 20, "nan", "15556201597"// 将结构体指针传入myPrintf(&obj);return 0;
}

结构的自引用

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

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

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};//结构体嵌套初始化

结构体内存对齐

结构体内存对齐规则

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

2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。

对齐数 = 编译器默认的一个对齐数 与 该成员大小的较小值。(gcc没有编译器对齐数,对齐数就是成员自身大小)

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

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

struct S1
{char c1;int i;char c2;
};struct S2
{char c1;char c2;int i;
};struct S3
{double d;char c;int i;
};printf("%d\n", sizeof(struct S1)); // 12
printf("%d\n", sizeof(struct S2)); // 8
printf("%d\n", sizeof(struct S3)); // 16

修改默认对齐数 #pragma pack(n) 

/* 修改编译器默认对齐数为4,请看结构体规则第2点 */
#pragma pack(4)
struct S
{int i;    // 0double d; // 4 8 ——> 4
};
#pragma pack()#pragma pack(1)
struct Sd
{char c;int i;char c1;
};
#pragma pack()printf("%d\n", sizeof(struct S));  // 12 (修改对齐数之前是16)
printf("%d\n", sizeof(struct Sd)); // 6

offsetof 求结构体成员相对于结构体开头的偏移量的宏。 

    
struct S1
{char c1;int i;char c2;
};struct S3
{double d;char c;int i;
};// S1
printf("%d\n", offsetof(struct S1, c1)); // 0
printf("%d\n", offsetof(struct S1, i));  // 4
printf("%d\n", offsetof(struct S1, c2)); // 8
// s3
printf("%d\n", offsetof(struct S3, d)); // 0
printf("%d\n", offsetof(struct S3, c)); // 8
printf("%d\n", offsetof(struct S3, i)); // 12

结构在对齐方式不合适的时候,我们可以自己更改默认对齐数。 

为什么存在内存对齐?

1. 平台原因(移植原因): 不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。

2. 性能原因: 数据结构(尤其是栈)应该尽可能地在自然边界上对齐。

原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问。

总体来说: 结构体的内存对齐是拿空间来换取时间的做法

结构体传参

函数传参的时候,参数是需要压栈,会有时间和空间上的系统开销。如果传递一个结构体对象的时候,结构体过大,参数压栈的的系统开销比较大,所以会导致性能的下降。

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

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

什么是位段

位段的声明和结构是类似的,有两个不同:
  1. 位段的成员必须是 int、unsigned int 或signed int 。
  1. 位段的成员名后边有一个冒号和一个数字。
struct A
{int _a : 2; int _b : 5;int _c : 10; int _d : 30; 
};

 A就是一个位段类型。

位段的内存分配规则

1. 位段的成员可以是 int unsigned int signed int 或者是 char (属于整形家族)类型

2. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的。

3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段。

4. int 位段被当成有符号数还是无符号数是不确定的。

5. 位段中最大位的数目不能确定。(16位机器最大16,32位机器最大32,写成27,在16位机器会出问题。

6. 位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。

7. 当一个结构包含两个位段,第二个位段成员比较大,无法容纳于第一个位段剩余的位时,是舍弃剩余的位还是利用,这是不确定的

struct A
{/* 上来二话不说先开辟4个字节空间,也就是 32个bit位 */int _a : 2;  // _a占2个bit位int _b : 5;  // _b占5个bit位int _c : 10; // _c占10个bit位/* 到这里发现 只剩 15 个bit位,不够存放_d的30个bit,那么会再次开辟4个字节用来存放 */int _d : 30; // _d占30个bit位
};printf("%d\n", sizeof(struct A)); // 8 未使用位段就是16个字节

位段的跨平台问题

  1. int 位段被当成有符号数还是无符号数是不确定的。
  2. 位段中最大位的数目不能确定。(16位机器最大16,32位机器最大32,写成27,在16位机
  3. 器会出问题。
  4. 位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。
  5. 当一个结构包含两个位段,第二个位段成员比较大,无法容纳于第一个位段剩余的位时,是舍弃剩余的位还是利用,这是不确定的

总结:跟结构相比,位段可以达到同样的效果,并且可以很好的节省空间,但是有跨平台的问题存在

位段的应用

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

pip list 和 conda list的区别

PS : 网上说conda activate了之后就可以随意pip了 可以conda和pip混用 但是安全起见还是尽量用pip 这样就算activate了&#xff0c;进入base虚拟环境了 conda与pip的区别 来源 Conda和pip通常被认为几乎完全相同。虽然这两个工具的某些功能重叠&#xff0c;但它们设计用于不…

硬件开发笔记(十二):RK3568底板电路电源模块和RTC模块原理图分析

若该文为原创文章&#xff0c;转载请注明原文出处 本文章博客地址&#xff1a;https://hpzwl.blog.csdn.net/article/details/134429973 红胖子网络科技博文大全&#xff1a;开发技术集合&#xff08;包含Qt实用技术、树莓派、三维、OpenCV、OpenGL、ffmpeg、OSG、单片机、软硬…

清华源Certificate verification failed解决办法

Certificate verification failed: The certificate is NOT trusted. The certificate chain uses expired certificate. Could not handshake: Error in the certificate verification. sudo update-ca-certificates

C++之type traits

目录 帮助类std::integral_constantbool_constant 主要类型分类std::is_voidis_null_pointeris_integralis_floating_pointis_arrayis_enumis_unionis_classis_functionis_pointeris_lvalue_referenceis_rvalue_referenceis_member_object_pointeris_member_function_pointer 组…

rpc依赖安装

依赖&#xff1a; 0、boost&#xff1a;用于实现多线程等&#xff1b; 1、protobuf&#xff1a;用于实现数据的序列化、反序列化&#xff0c;也用于定义和生成rpc数据及接口&#xff1b; 2、libevent&#xff1a;用于实现基于IO多路复用机制的网络事件循环。 其实可以直接用包…

JAVA sql 查询

-- 1. 查询员工表所有数据&#xff0c;并说明使用*的缺点 SELECT * from employees -- *号查询效率低 -- 2. 查询所员工的 email 全名,公司 email 统一以 "qq.com " 结尾. SELECT email from employees WHERE email like "%qq.com" -- 3. 打印公司里…

庖丁解牛:NIO核心概念与机制详解 06 _ 连网和异步 I/O

文章目录 Pre概述异步 I/OSelectors打开一个 ServerSocketChannel选择键内部循环监听新连接接受新的连接删除处理过的 SelectionKey传入的 I/O回到主循环 Pre 庖丁解牛&#xff1a;NIO核心概念与机制详解 01 庖丁解牛&#xff1a;NIO核心概念与机制详解 02 _ 缓冲区的细节实现…

Go基础面经大全(持续补充中)

Go基础 1. 基础特性 Go的优势 天生支持并发&#xff0c;性能高。 单一的标准代码格式&#xff0c;比其他语言更具可读性。 自动垃圾收集机制比Java和Python更有效&#xff0c;因为它与程序同时执行。 Go数据类型 int, string, float, bool, array, slice, map, channel, p…

如何将本地Portainer管理界面结合cpolar内网穿透工具实现远程浏览器访问

文章目录 前言1. 部署Portainer2. 本地访问Portainer3. Linux 安装cpolar4. 配置Portainer 公网访问地址5. 公网远程访问Portainer6. 固定Portainer公网地址 前言 Portainer 是一个轻量级的容器管理工具&#xff0c;可以通过 Web 界面对 Docker 容器进行管理和监控。它提供了可…

Android修行手册-POI操作中文API文档

Unity3D特效百例案例项目实战源码Android-Unity实战问题汇总游戏脚本-辅助自动化Android控件全解手册再战Android系列Scratch编程案例软考全系列Unity3D学习专栏蓝桥系列ChatGPT和AIGC &#x1f449;关于作者 专注于Android/Unity和各种游戏开发技巧&#xff0c;以及各种资源分…

LLM模型-讯飞星火与百度文心api调用

spark-wenxin 1-讯飞星火1_1-SparkApi.py1_2- Chat_spark.py1_3-调用api 2-百度文心2_1.code 3-两者之间比较与openai 1-讯飞星火 进入讯飞官网进行创建应用&#xff0c;获取相关密钥APPID&#xff0c;APISecret&#xff0c;APIKey&#xff0c;选择最新版本 首次调用讯飞官方a…

redis三种集群方式

redis有三种集群方式&#xff1a;主从复制&#xff0c;哨兵模式和集群。 1.主从复制 主从复制原理&#xff1a; 从服务器连接主服务器&#xff0c;发送SYNC命令&#xff1b; 主服务器接收到SYNC命名后&#xff0c;开始执行BGSAVE命令生成RDB文件并使用缓冲区记录此后执行的所…

Git的简单使用

Git的简介 Git是一个分布式版本控制工具&#xff0c;通常用来对软件开发过程中的源代码文件进行管理。通过Git仓库来存储和管理这些文件&#xff0c;Git仓库分为两种&#xff0c;一个是本地仓库&#xff1a;开发人员自己电脑上的Git仓库&#xff0c;一个是远程仓库&#xff1a;…

centos FreeXL源码编译

安装、编译 https://www.gaia-gis.it/gaia-sins/freexl-2.0.0.zip wget https://www.gaia-gis.it/gaia-sins/freexl-2.0.0.zip#centos yum install -y minizip minizip-devel expat-devel ./configure --prefix/usr/local/freexl-2.0.0#Ubuntu apt install libminizip libmin…

学习c#的第十六天

目录 C# 正则表达式 定义正则表达式 字符转义 字符类 定位点 分组构造 Lookaround 概览 数量词 反向引用构造 替换构造 替代 正则表达式选项 其他构造 Regex 类 代码示例 实例 1 实例 2 实例 3 C# 正则表达式 正则表达式 是一种匹配输入文本的模式。.Net 框…

PHP比较漏洞, 代码审计, 字符串数字, “0e“字符串, 布尔值比较, 极限值, switch比较, in_array比较

PHP比较漏洞 一, 字符串与数字比较,0e’字符串比较 在 PHP 中处理字符串和数字比较时需要小心谨慎的重要性&#xff0c;尤其是在安全敏感的应用中。 在某些情况下&#xff0c;PHP 的类型强制转换可能会导致意外和潜在的安全漏洞。 var_dump("admin" 0); //true …

RecyclerView嵌套布局,导致RecyclerView复用失效 解决

前言&#xff1a;使用NestedScrollView嵌套RecyclerView。 解决步骤一&#xff1a;固定高度 NestedScrollView嵌套RecyclerView时&#xff0c;RecyclerView的高度是无限大&#xff0c;所以要将RecyclerView设置固定高度。在代码中固定的&#xff0c;灵活度更高。 binding.nest…

键盘方向键移动当前选中的table单元格,并可以输入内容

有类似于这样的表格&#xff0c;用的<table>标签。原本要在单元格的文本框里面输入内容&#xff0c;需要用鼠标一个一个去点以获取焦点&#xff0c;现在需要不用鼠标选中&#xff0c;直接用键盘的上下左右来移动当前正在输入的单元格文本框。 const currentCell React.u…

【数据结构初阶】单链表SLlist

描述 不同于顺序表&#xff0c;顺序表的数据是存储在一个连续的空间里的 而链表它是链接起来的结构体地址。 所以我们不用像顺序表一样先创建一块空间出来&#xff0c;而是创建一个能存数据节点和节点与下一个节点之间的连接&#xff1b; 所以&#xff1a;“一个能存数据节点…

词向量笔记

一、词向量的世界排名比赛MTEB MTEB: Massive Text Embedding Benchmark 二、词向量的维数 词向量的维数d1一般取20~500之间