【C语言】自定义类型:结构体【结构体内存具详细】,枚举,联合

 

目录

一、结构体

1.结构的声明

 2.特殊的声明

3.结构的自引用

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

 5.结构体内存对齐(重点来了)

6.为什么会存在内存对齐 

7.修改默认对齐数

8.结构体传参

二、位段

1.什么是位段

2.位段的内存分配

3.位段的跨平台问题

三、枚举

1.枚举类型的定义

2.枚举的优点

3.枚举的使用

四、联合(共用体)


一、结构体

在此之前简单地介绍了结构体的使用初阶结构体

首先 结构结构是一些值得集合,这些值称为成员变量。结构的每个成员可以是不同类型的变量。

1.结构的声明

//结构体的声明
struct tag 
{member - list;//成员列表
}variable-list;//变量列表

【例如】

struct Stu
{char name[20];int age;char sex[5];char id[20];
};

 2.特殊的声明

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

如 匿名结构体类型

//匿名结构体类型
struct 
{int a;char b;float c;
}x;
struct 
{int a;char b;float c;
}a[20],*p;
//注意 不能这样写 p = &x;//编译器会把上述当成两个不同的结构体,所以是非法的

3.结构的自引用

就是结构体中含有一个类型为结构体本身的成员,如果学过数据结构的话,可能会知道链表中的node节点 就用到了结构体的自引用。

结构体的自引用方式

struct Node 
{int data;//数据域struct Node* next;//指针域,指向下一个节点
};
//另一种写法,使用类型重命名
typedef struct Node
{int data;struct Node* next;
}Node;//这样就可以 当写 struct Node 可以写为 Node

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

(1)声明类型的同时定义变量p1

//结构体变量的定义和初始化
struct Point
{int x;int y;
}p1;//声明类型的同时定义变量p1

(2)初始化,定义变量的同时赋值

struct Point p3 = {x,y};struct Stu 
{char name[15];int age;
};
struct Stu s = {"zhangsan",20};

(3)结构体嵌套初始化

struct Stu 
{int data;struct Point p;struct Node* next;
}n1 = { 10,{4,5},NULL };//结构体嵌套初始化

 5.结构体内存对齐(重点来了)

先来个问题,如何计算结构体的大小呢?这个问题就涉及了结构体内存对齐

看下方代码,并试着计算这俩个结构体分别是多大

struct S1 
{char c1;int a;char c2;
};
struct S2
{char c1;char c2;int a;
};

如果不了解结构体内存对齐的话,可能会认为这两个结构体不一样大吗?

当我们试着打印这两个结构体的大小

这时我们发现这两个结构体的大小不一样大,这是为什么呢?

看完接下来的结构体内存对齐,就明白了。

重点

要计算结构体的大小,那么我们就要掌握下方的

结构体对齐规则:

  1. 结构体的第一个成员  在 与结构体变量偏移量为0的地址处
  2. 其他成员变量要对齐到某个数字(即对齐数)的整数倍的地址处。
    对齐数 = 编译器默认的一个对齐数 与 该成员大小的 较小值(就是编译器的对齐数 与成员变量 这两个中较小的一方 )
    注意:VS中对齐数默认值是8
  3. 结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍
  4. 如果有嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍

根据上述结构体对齐规则,我们来分析下方代码

【1】

struct S1 
{char c1;//所占大小为1int a;//所占大小为4char c2;//所占大小为1
};
//该结构体的总体大小为?

该图解具详细 

接着来分析   struct S2

【2】

struct S2
{char c1;//所占大小为1char c2;//所占大小为1int a;//所占大小为4
};//该结构体总体大小为?

图解具详细:

有了上述两个 我们接着再来看两个,巩固一下结构体内存

【3】根据结构体的内存对齐计算结构体的大小

struct S3 
{double d;char c;int i;
};

图解具详细:

【4】 计算结构体S4的大小

struct S3 
{double d;char c;int i;
};
struct S4 
{char z;struct S3 s3;double d;
};

6.为什么会存在内存对齐 

简单来说就是

  1. 平台原因 某些硬件导致的原因
  2. 性能原因 访问一些内存时,在访问为未对齐的内存,可能会访问两次内存,而访问内存对齐的内存时 ,只要访问一次即可 

所以,对于结构体的内存对齐 是 通过 空间 来换取 时间 的做法

7.修改默认对齐数

我们可能之前会看到#pragma 这个预处理指令

#pragma 可以修改默认对齐数

#include<stdio.h>
#pragma pack(1)//对齐数修改为4
struct S1 
{char c1;int a;char c2;
};
int main() 
{printf("%d",sizeof(struct S1));
}

这样经过修改默认对齐数,改为1。这样计算出来的结构体大小就是6了。

有了这个pragma,当结构体对齐方式不合适的时候我们就可以修改默认对齐数,进而来调整结构体的大小。

8.结构体传参

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

注意: 因为函数传参的时候要压栈,会有时间和空间上的系统开销。而传结构体的地址的时候与会减少系统的开销。

二、位段

1.什么是位段

位段的声明和结构体是类似的,但有两点不同

  1. 位段的成员必须是int、unsigned int 、
  2. 位段的成员名后边有一个冒号和一个数字。

【例】

struct A {int a : 2;int b : 3;int c : 4;int d : 5;
};

2.位段的内存分配

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

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

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

3.位段的跨平台问题

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

三、枚举

1.枚举类型的定义

枚举就是列举

enum Color 
{RED,GREEN,BLUE
};
enum Sex 
{MALE,FEMALE,SECRET
};

enum Sex,enum Color都是枚举类型,{ }中的内容就是枚举类型可能取的值,也叫枚举常量

有些可以有值得,默认从0开始,一次递增1,定义的时候可以赋值

enum Color 
{RED=1,GREEN=2,BLUE=3
};

2.枚举的优点

1. 增加代码的可读性和可维护性
2. 和#define定义的标识符比较枚举有类型检查,更加严谨。
3. 防止了命名污染(封装)
4. 便于调试
5. 使用方便,一次可以定义多个常量

3.枚举的使用

enum Color
{RED = 1,GREEN = 2,BLUE = 3
};
enum Color clr = GREEN;//这样不会出现类型的差异
clr = 5;

四、联合(共用体)

联合也是一种特殊的自定义类型,这种类型定义的变量也包含一系列的成员,特征是这些成员共用同一块空间(所以联合也叫共用体)

#include<stdio.h>
union Un
{int i;char c;
};
union Un un;
int main()
{un.i = 0x11223344;un.c = 0x55;printf("%d\n",&(un.i));printf("%d\n",&(un.c));printf("%x\n",un.i);return 0;
}

从第三个的打印结果我们可以看出来 联合体中的成员就是共用同一块内存空间。

来一个笔试题

判断当前计算机的大小端存储

//判断大小端的存储
#include<stdio.h>
int check_sys() 
{int i = 1;return *((char*)&i);// 00 00 00 01 大端// 01 00 00 00 小端//地址 低---->高
}
int main() 
{int ret = check_sys();if (ret == 1)printf("小端\n");elseprintf("大端\n");return 0;
}

上述代码就是我们通过取地址的方式来判断的

接下来我们来使用联合体来判断判断当前机器的字节序

//使用联合体来判断当前机器的字节序
#include<stdio.h>
int check_sys() 
{union {int i;char c;}un;un.i = 1;return un.c;//这里的联合体共用是同一块内存空间
}
int main() 
{int ret = check_sys();if (ret == 1)printf("小端\n");elseprintf("大端\n");return 0;
}

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

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

相关文章

nodejs项目实战(带源码)

nodejs项目实战 主要实现功能用户模块文章分类模块文章模块核心代码 数据库完整代码 主要实现功能 本项只适合新手&#xff0c;是一个接口类的项目&#xff0c;主要涉及一些增删改查功能以及三方包的使用&#xff0c;主要包括用node实现写用户登录注册&#xff0c;添加删除文章…

win系统环境搭建(三)——Windows安装maven

windows环境搭建专栏&#x1f517;点击跳转 win系统环境搭建&#xff08;三&#xff09;——Windows安装maven 本系列windows环境搭建开始讲解如何给win系统搭建环境&#xff0c;本人所用系统是腾讯云服务器的Windows Server 2022&#xff0c;你可以理解成就是你用的windows10…

基于Spring Boot+ Vue的健身房管理系统与实现

小熊学Java全能学面试指南&#xff1a;https://javaxiaobear.cn 摘要 随着健身行业的快速发展&#xff0c;健身房管理系统成为了提高管理效率和用户体验的重要工具。本论文旨在设计与实现一种基于前后端分离的健身房管理系统&#xff0c;通过前后端分离的架构模式&#xff0c;…

ThreadLocal线程局部变量

1.原理 ThreadLocal是用来保存当前线程数据的&#xff0c;每一个线程的内部都有一个ThreadLocalMap&#xff0c;当前这个map中存储了以当前ThreadLocal作键&#xff0c;具体的数据作值的一个个Entry对象。 为什么非得以ThreadLocal对象作键呢&#xff1f;因为一个线程可能使用了…

若依前端使用

初始化页面时&#xff0c;路由上加参数 多个菜单对应一个页面&#xff0c;默认查询的数据状态不一样 vue 页面上 通过 debugger; 查看所有的参数&#xff0c; 最后取到了

网络基础(一)

网络基础&#xff08;一&#xff09; 在本篇文章中&#xff0c;我会详细介绍网络的一些基础知识&#xff0c;之后也会有持续的更新来帮助大家学习网络&#xff0c;希望大家可以通过博客学到更多的知识&#xff01; 了解网络 去百度查询可以知道&#xff0c;网络是由若干节点和…

Unity SteamVR 开发教程:用摇杆/触摸板控制人物持续移动(2.x 以上版本)

文章目录 &#x1f4d5;教程说明&#x1f4d5;场景搭建&#x1f4d5;创建移动的动作&#x1f4d5;移动脚本⭐移动⭐实时调整 CharacterController 的高度 &#x1f4d5;取消手部和 CharacterController 的碰撞 持续移动是 VR 开发中的一个常用功能。一般是用户推动手柄摇杆&…

Netty2

文章目录 Netty2Netty入站与出站机制Netty的handler链的调用机制 Netty2 Netty入站与出站机制 基本说明&#xff1a; 1&#xff09;netty的组件设计&#xff1a;Netty的主要组件有Channel&#xff0c;EventLoop&#xff0c;ChannelFuture&#xff0c;ChannelHandler&#xff…

SaaS架构C/S检验科LIS系统源码: 检验申请、标本编号、联机采集

适用于医院检验科实际需要的LIS管理系统, 实现检验业务全流程的计算机管理。从检验申请、标本编号、联机采集、中文报告单的生成与打印、质控图的绘制和数据的检索与备份。通过将所有仪器自身提供的端口与科室LIS系统中的工作站点连接,实现与医院HIS系统的对接。 通过门诊医生和…

如何使用ArcGIS中的Arcmap进行矢量和栅格数据裁剪?

在地理信息系统(GIS)中&#xff0c;我们经常需要处理各种空间数据&#xff0c;而矢量和栅格数据是最常见的两种数据类型。有时候&#xff0c;我们需要对数据进行裁剪&#xff0c;以提取出我们需要的特定区域的数据。本文将介绍如何使用ArcGIS中的Arcmap软件对矢量和栅格数据进行…

选择渲染农场的几个标准

随着电影、电视剧等影视作品的制作越来越依赖于计算机特效&#xff0c;渲染农场的使用也变得越来越普遍。渲染农场是一种利用大量计算机图形处理器&#xff08;GPU&#xff09;来加速渲染过程的服务。在选择渲染农场时&#xff0c;有几个标准可以帮助您确定哪个农场是适合您的项…

Unity3D C# 反射与特性的配合使用

需求分析 情况&#xff1a; 假如我们是一个动物园的管理员&#xff0c;我们需要统计园内的所有动物和动物的行为。 举例&#xff1a; 现在园区内有猫、狗和鸡。猫对应的行为是喵喵喵和卖萌&#xff0c;狗对应狗吠和干饭&#xff0c;鸡对应篮球和打鸣那么这时候我要统计这些&a…