当前位置: 首页 > news >正文

自定义类型:结构体

自定义类型:结构体

  • 一:引入
  • 二:结构体类型的声明
    • 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;
}

相关文章:

自定义类型:结构体

自定义类型&#xff1a;结构体 一&#xff1a;引入二&#xff1a;结构体类型的声明1&#xff1a;正常声明2&#xff1a;特殊声明 三&#xff1a;结构体变量的创建和初始化1:结构体变量的创建2&#xff1a;结构体变量的初始化 三&#xff1a;结构体访问操作符四&#xff1a;结构…...

postman如何设置才能SwitchHosts切换host无缓存请求到指定ip服务

开发测试中,遇到多版本同域名的服务使用postman进行测试,一般会搭配SwitchHosts切换host类似工具进行请求,postman缓存比较重,如何做到无缓存请求呢,下面简单记录一下如何实现 首先要知道如何当前请求服务的ip是哪个 打开postman 依次点击/menu/view/show postman console 就…...

LeetCode LCR 103. 零钱兑换【完全背包,恰好装满背包的最小问题】中等

本文属于「征服LeetCode」系列文章之一&#xff0c;这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁&#xff0c;本系列将至少持续到刷完所有无锁题之日为止&#xff1b;由于LeetCode还在不断地创建新题&#xff0c;本系列的终止日期可能是永远。在这一系列刷题文章…...

竞赛 基于深度学习的人脸专注度检测计算系统 - 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 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; 基于深度学习的人脸专注度…...

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的主从复制&#xff1f; MySQL的主从复制是一种常见的数据库复制技术&#xff0c;用于将一个数据库服务器&#xff08;称为主…...

ChatGLM 大模型炼丹手册-理论篇

序言一)大还丹的崛起 在修真界,人们一直渴望拥有一种神奇的「万能型丹药」,可包治百病。 但遗憾的是,在很长的一段时间里,炼丹师们只能对症炼药。每一枚丹药,都是特效药,专治一种病。这样就导致,每遇到一个新的问题,都需要针对性的炼制,炼丹师们苦不堪言,修真者们吐…...

Spring Boot集成Redis实现数据缓存

&#x1f33f;欢迎来到衍生星球的CSDN博文&#x1f33f; &#x1f341;本文主要学习Spring Boot集成Redis实现数据缓存 &#x1f341; &#x1f331;我是衍生星球&#xff0c;一个从事集成开发的打工人&#x1f331; ⭐️喜欢的朋友可以关注一下&#x1faf0;&#x1faf0;&…...

CentOS 7 安装Libevent

CentOS 7 安装Libevent 1.下载安装包 新版本是libevent-2.1.12-stable.tar.gz。&#xff08;如果你的系统已经安装了libevent&#xff0c;可以不用安装&#xff09; 官网&#xff1a;http://www.monkey.org/~provos/libevent/ 2.创建目录 # mkdir libevent-stable 3.解压 …...

线性代数的本质——几何角度理解

B站网课来自 3Blue1Brown的翻译版&#xff0c;看完醍醐灌顶&#xff0c;强烈推荐&#xff1a; 线性代数的本质 本课程从几何的角度翻译了线代中各种核心的概念及性质&#xff0c;对做题和练习效果有实质性的提高&#xff0c;下面博主来总结一下自己的理解 1.向量的本质 在物…...

SSH key 运作方式

1、本地创建SSH key pairs 2、把public key上传到网站服务器&#xff08;如GitHub 3、当使用ssh方式连接时 本地SSH client向远端请求ssh连接远端发来random data要求加密本地ssh client用private key加密&#xff0c;把加密的data发送过去&#xff08;不发送private key远端接…...

【基于MBD开发模式的matlab持续集成(一)】

基于MBD开发模式的matlab持续集成 引言 或许是感受到行业内卷的愈加激烈&#xff0c;在传统制造和高新技术相结合的新能源领域对软件工程开发的要求也愈加提高&#xff0c;尤其在互联网已经大行 其道的敏捷开发&#xff0c;便顺其自然的被新能源的老板们所看重。 概述 本文…...

Linux学习记录——이십팔 网络基础(1)

文章目录 1、了解2、网络协议栈3、TCP/IP模型4、网络传输1、同一局域网&#xff08;子网&#xff09;2、局域网通信原理3、跨一个路由器的两个子网4、其它 详细的网络发展历史就不写了 1、了解 为什么会出现网络&#xff1f;一开始多个计算机之间想要共享文件&#xff0c;就得…...

CSS动效合集之实现气泡发散动画

前言 &#x1f44f;CSS动效合集之实现气泡发散动画&#xff0c;速速来Get吧~ &#x1f947;文末分享源代码。记得点赞关注收藏&#xff01; 1.实现效果 2.实现步骤 定义一个数组bubbles&#xff0c;用来存储气泡列表的基本新&#xff0c;w表示宽高&#xff0c;x表示绝对定位…...

六、串口通信

六、串口通信 串口接口介绍使用串口向电脑发送数据电脑发送数据控制LED灯 串口接口介绍 SBUF是串口数据缓存器&#xff0c;物理上是两个独立的寄存器&#xff0c;但占用相同的地址。写操作时&#xff0c;写入的是发送寄存器&#xff1b;读操作时&#xff0c;读出的是接收寄存器…...

如何将 JavaScript Excel XLSX 查看器添加到Web应用程序

在 JavaScript 中创建 Excel 查看器可能是一项艰巨的任务&#xff0c;但使用 SpreadJS JavaScript 电子表格&#xff0c;创建过程要简单得多。在本教程博客中&#xff0c;我们将向您展示如何使用 SpreadJS 的强大功能来创建一个查看器&#xff0c;该查看器允许您在 Web 浏览器中…...

网安周报|CISA发布增强开源安全性的计划

1、CISA发布增强开源安全性的计划 美国一家领先的安全机构发布了一项期待已久的计划&#xff0c;详细说明了它将如何增强联邦政府和整个生态系统的开源安全性。美国网络安全和基础设施安全局&#xff08;CISA&#xff09;开源软件安全路线图在安全开源峰会上发布。据估计&#…...

使用 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&#xff1a;是multithread-static version&#xff0c;是多线程静态版本的意思&#xff0c;项目会使用运行时库的多线程静态版本&#xff0c;编译器会将LIBCMT.lib放入.obj文件中&#xff0c;以便链接器使用LIBCMT.lib解析外部符号&#xff1b;/MTd&#xff…...

Vue3函数式编程

文章目录 前言一、三种编程风格1.template2.jsx/tsx3.函数式编写风格 二、函数式编程1.使用场景2.参数3.例子3.render渲染函数 总结 前言 本文主要记录vue3中的函数式编程以及其他编程风格的简介 一、三种编程风格 1.template Vue 使用一种基于 HTML 的模板语法&#xff0c;…...

Android Wi-Fi 连接失败日志分析

1. Android wifi 关键日志总结 (1) Wi-Fi 断开 (CTRL-EVENT-DISCONNECTED reason3) 日志相关部分&#xff1a; 06-05 10:48:40.987 943 943 I wpa_supplicant: wlan0: CTRL-EVENT-DISCONNECTED bssid44:9b:c1:57:a8:90 reason3 locally_generated1解析&#xff1a; CTR…...

shell脚本--常见案例

1、自动备份文件或目录 2、批量重命名文件 3、查找并删除指定名称的文件&#xff1a; 4、批量删除文件 5、查找并替换文件内容 6、批量创建文件 7、创建文件夹并移动文件 8、在文件夹中查找文件...

PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建

制造业采购供应链管理是企业运营的核心环节&#xff0c;供应链协同管理在供应链上下游企业之间建立紧密的合作关系&#xff0c;通过信息共享、资源整合、业务协同等方式&#xff0c;实现供应链的全面管理和优化&#xff0c;提高供应链的效率和透明度&#xff0c;降低供应链的成…...

高等数学(下)题型笔记(八)空间解析几何与向量代数

目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...

【算法训练营Day07】字符串part1

文章目录 反转字符串反转字符串II替换数字 反转字符串 题目链接&#xff1a;344. 反转字符串 双指针法&#xff0c;两个指针的元素直接调转即可 class Solution {public void reverseString(char[] s) {int head 0;int end s.length - 1;while(head < end) {char temp …...

【Java学习笔记】BigInteger 和 BigDecimal 类

BigInteger 和 BigDecimal 类 二者共有的常见方法 方法功能add加subtract减multiply乘divide除 注意点&#xff1a;传参类型必须是类对象 一、BigInteger 1. 作用&#xff1a;适合保存比较大的整型数 2. 使用说明 创建BigInteger对象 传入字符串 3. 代码示例 import j…...

IP如何挑?2025年海外专线IP如何购买?

你花了时间和预算买了IP&#xff0c;结果IP质量不佳&#xff0c;项目效率低下不说&#xff0c;还可能带来莫名的网络问题&#xff0c;是不是太闹心了&#xff1f;尤其是在面对海外专线IP时&#xff0c;到底怎么才能买到适合自己的呢&#xff1f;所以&#xff0c;挑IP绝对是个技…...

Redis:现代应用开发的高效内存数据存储利器

一、Redis的起源与发展 Redis最初由意大利程序员Salvatore Sanfilippo在2009年开发&#xff0c;其初衷是为了满足他自己的一个项目需求&#xff0c;即需要一个高性能的键值存储系统来解决传统数据库在高并发场景下的性能瓶颈。随着项目的开源&#xff0c;Redis凭借其简单易用、…...

省略号和可变参数模板

本文主要介绍如何展开可变参数的参数包 1.C语言的va_list展开可变参数 #include <iostream> #include <cstdarg>void printNumbers(int count, ...) {// 声明va_list类型的变量va_list args;// 使用va_start将可变参数写入变量argsva_start(args, count);for (in…...

xmind转换为markdown

文章目录 解锁思维导图新姿势&#xff1a;将XMind转为结构化Markdown 一、认识Xmind结构二、核心转换流程详解1.解压XMind文件&#xff08;ZIP处理&#xff09;2.解析JSON数据结构3&#xff1a;递归转换树形结构4&#xff1a;Markdown层级生成逻辑 三、完整代码 解锁思维导图新…...