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

C语言中的自定义类型 —— 结构体.位段.联合体和枚举

自定义类型

  • 1. 前言
  • 2. 结构体
    • 2.1 结构体的声明
    • 2.2 结构体变量的定义和初始化
    • 2.3 结构体的特殊声明
    • 2.4 结构体的自引用
    • 2.5 结构体的内存对齐
    • 2.6 修改默认对齐数
    • 2.7 结构体传参
  • 3. 位段
  • 4. 联合体
  • 5. 枚举
  • 6. 结言

1. 前言

在C语言中已经为用过户提供了内置类型,如:char,short,int,long等等,但是仅仅只有这几种类型是远远不够的,并不能满足用户的需求,当用户描述的对象很复杂(属性较多)时,如一个学生,一本书,这时单一的内置类型是不行的。在前面的数据结构中已经体会到了单一内置类型的不足。而C语言为了解决这个问题,增加了结构体这种自定义的数据类型(何为自定义类型 —— 根据需要,用户自己来设计的一种类型),让用户可以自己创造适合对象的类型。

2. 结构体

2.1 结构体的声明

使用结构体时需要用到结构体的关键字 —— struct。什么是结构呢?结构是一些值的集合,这些值被称作成员变量,结构的每个成员可以是不同类型的的变量,如:标量,数组,指针,甚至是其它结构体。那么如何去声明一个结构体?如下所示:

struct tag
{member-list;
}variable-list;

tag —— 结构体标签的名字,根据用户的需要去设计,与变量的变量名设计一致
member-list —— 结构体的成员列表,存在一个或多个成员
variable —— 结构体的变量列表,可以省略不写
注意:结构体末尾存在一个分号

下面就来声明一个书这样的结构体类型 —— 属性有书名,作者,价格等等:

struct book
{char name[20];   // 书名char author[20]; // 作者double price;    // 价格
};

这样我们就声明了一个书结构体类型。

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

有了结构体的声明之后,就可以使用这个结构体的类型了,如何使用呢?—— 结构体类型就是一个模具,声明出了结构体类型后,就可以定义出结构体类型的变量,定义方式:struct 结构体名字 + 变量名。如下所示:

struct book book1;
struct book book2;

book1 和 book2 就是结构体变量,结构体变量可以是局部变量也可以是全局变量。全局变量不仅仅可以直接创建在局部域当中,也可以在结构体声明的同时创建结构体变量,如下所示:

struct book
{char name[20];   // 书名char author[20]; // 作者double price;    // 价格
}book5, book6; // 全局变量//全局变量
struct book book3;
struct book book4;int main()
{//局部变量struct book book1;struct book book2;return 0;
}

变量定义完毕后,需要对结构体变量里的成员初始化,怎么初始化呢?
首先初始化是在一个大括号里,初始化的顺序可以与结构体的声明中成员的顺序不一致,但是最好是一致的,不一致的需要额外的操作。
具体初始化方式如下:

初始化顺序与结构体声明中的成员的顺序一致:

struct book book1 = { "老鹰抓小鸡", "埃伊蟹黄面", 88.8 };
struct book book2 = { "小鸡自卫战", "埃伊蟹黄面", 99.9 };

初始化顺序与结构体声明中的成员的顺序不一致:

struct book book3 = { .author = "埃伊蟹黄面", .name = "小鸡吃老鹰", .price = 77.7 };
struct book book4 = { .price = 109.8, .name = "小鸡变老鹰", .author = "埃伊蟹黄面" };

需要用到 . 操作符来访问具体的成员变量。
之前有提到过,结构体的成员类型不仅可以是基础的内置类型,也可以是结构体类型,那么当结构体中存在另一个结构体类型时,又该怎么初始化呢?如下所示:

struct other
{char character;int rint;float rfloat;
};struct book
{char name[20];   char author[20]; struct other ther;double price;
};int main()
{struct book book5 = { "ppt", "ptr", { 'G', 68, 53.2 }, 9.9 };return 0;
}

结构体类型的成员变量的初始化也需要用另外的大括号括起。既然初始化完毕,我们可以尝试将这些初始化值打印出来,但是结构体中有这么多的成员变量怎么找到要打印的变量呢?可以通过 . 操作符来访问具体的成员变量 ,访问方式为:结构体的名字.要访问的变量的名字,如下所示:

struct book book5 = { "ppt", "ptr", { 'G', 68, 53.2 }, 9.9 };//单独访问
printf("%s\t", book5.name);
printf("%s\t", book5.author);
printf("%.2f\t", book5.price);
printf("%c\t", book5.ther.character);
printf("%d\t", book5.ther.rint);
printf("%.2f\n", book5.ther.rfloat);//一起访问
printf("%s\t%s\t%.2f\n", book5.name, book5.author, book5.price);
printf("%c\t%d\t%.2f", book5.ther.character, book5.ther.rint, book5.ther.rfloat);

运行结果:

在这里插入图片描述
初始化的值并不是一成不变,在初始化后,可以对自己想要改变的变量的值进行修改,具体如下:

struct book book5 = { "ppt", "ptr", { 'G', 68, 53.2 }, 9.9 };
book5.price = 99.9;
book5.ther.character = 'H';
strcpy(book5.name, "小鸡的独立");

修改后,代码的运行结果:
在这里插入图片描述
由上图可知,结构体中变量的值被修改了。

2.3 结构体的特殊声明

在声明结构体的时候,可以不完全的声明,之前的结构体的声明是完全的结构体声明。不完全的结构体的声明是怎样的呢?如下代码所示:

struct
{int id;char name[20];int age;double high;
};

上面的代码就是不完全的结构体的声明,省略了结构体的名字,所以这种结构体也称为匿名结构体。将结构体写成匿名结构体后,结构体变量的创建的就只能在声明的同时创建,因为这个结构体没有标签(名字)呀,怎么在局部域中创建?具体创建方式如下所示:

struct
{int id;char name[20];int age;double high;
}stu1, stu2;

对于匿名结构体类型,如果没有对结构体类型重命名的话,基本上只能使用一次

2.4 结构体的自引用

结构体的自引用是什么意思呢?—— 结构体中包含一个或多个类型为该结构体本身类型的成员变量。数据结构中经常用到,如单链表的结点的结构体:

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

上面的就是结构体的自引用。在结构体的自引用当中可以使用typedef关键字对结构体进行重命名,如下所示:

typedef struct SListNode
{int data;struct SListNode* next;
}SLNode;

之后SLNode就相当于是struct SListNode,既然如此,能不能将结构体内部的struct SListNode* 替换成 SLNode* 呢?不能!因为SLNode是对前面的结构体类型重命名产生的,它是后于结构体类型创建的。

2.5 结构体的内存对齐

结构体既然是个自定义类型,那么该类型的内存大小又是多少呢?不同的结构体类型中的成员变量的类型不同,成员变量的个数不同会对结构体是内存大小产生影响,而要想知道怎么不通过sizeof来计算结构体类型的大小,就得要了解结构体内存对齐
在介绍结构体内存对齐前,来计算以下两个结构体的大小:
在这里插入图片描述
在不了解结构体内存对齐之前,计算的方法可能是这样:对于Test1结构体,第一个成员类型为char,所以该成员变量的内存大小为1个字节,以此类推,所以Test1结构体的内存大小一共为6个字节;对于Test2结构体,第一个成员类型为char,所以该成员变量的内存大小为1个字节,以此类推,所以Test2结构体的内存大小一共为6个字节。那么这样的计算逻辑有没有问题呢?接下来用sizoef来计算两个结构体类型的大小:

在这里插入图片描述
为什么得到的结果不是两个6,而是Test1的内存的大小为8个字节,Test2的内存大小为12个字节呢?因为结构体的成员在存储的时候会存在内存对齐现象。该现象涉及到了结构体当中的偏移量,何为偏移量,距离起始位置相差多少个字节就为多少偏移量,具体如下所示:
在这里插入图片描述
明白了偏移量是什么后。接下来开始介绍结构体内存对齐的规则:

1. 结构体的第一个成员变量对齐到和结构体变量的起始位置偏移量为0的位地址
2. 其它的结构体成员变量要对齐到某个数字(对齐数)的整数倍地址处
对齐数 = 编译器默认的一个对齐数 与 该成员变量大小的较小值
在编译器中,vs编译器中默认对齐数为 8
Linux中 gcc 没有默认对齐数,对齐数就是成员自身类型的大小
3. 结构体总大小为最大对齐数(结构体中每个成员变量的对齐数中的最大值)的整数倍
4. 如果嵌套了结构体的情况,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是最大对齐数(含嵌套结构体中的成员的对齐数)的整数倍

知道了结构体内存对齐的规则之后,接下来根据该规则计算先前的两个结构体类型的大小:

Test1结构体的内存大小计算:
在这里插入图片描述
Test2结构体的内存大小计算:

在这里插入图片描述
当要计算的结构体中又嵌套了一个结构体呢?这该如何计算?接下来就来详细的计算嵌套结构体类型的内存大小:

在这里插入图片描述

接下来用sizeof来计算Test2结构体的内存大小:

在这里插入图片描述
了解了内存对齐的运用之后,接下来思考为什么会存在内存对齐呢?有以下几个原因:

1. 平台原因
不是所有的硬件平台都能访问任意地址上的任意数据;某些硬件平台只能在某些地址处取得某些特定类型的数据,否则会抛出硬件异常。
2. 性能原因
数据结构应该尽可能地在自然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要做两次内存访问;而对齐的内存访问仅需要一次访问

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

在前面又计算过两个成员类型相同,成员个数相同的结构体,但是计算的结果却不一样,如下图所示:
在这里插入图片描述
这表明了在设计结构体的时候,让占用空间小的成员尽量集中在一起,这样就可以最大限度的节省空间

2.6 修改默认对齐数

其实vs编译器的默认对齐数8是可以修改的,#pragma这个预处理指令就可以修改编译器的默认对齐数。具体修改方法如下:

#pragma pack(2) // 设置默认对齐数为2
struct Test3
{char charc1;int rint;char charc2;
};
#pragma pack() // 取消设置的默认对齐数

下面与未修改默认对齐数的结构体的内存大小进行比较:
在这里插入图片描述
当结构体在对齐方式不合适的时候,可以自己更改默认对齐数。

2.7 结构体传参

结构体既然作为自定义类型,自然可以用作函数的参数。它与内置类型传参的一致,又传值传参和传址传参。下面来分别写成结构体的传值传参和传址传参。

传值传参:

在这里插入图片描述
若想要访问结构体对象中的成员,可以通过 . 操作符 来访问。

传址传参:

在这里插入图片描述

若想要访问结构体对象中的成员,可以通过 ->操作符 来访问。

无论是传值传参还是传址传参,都满足我们的要求,那么到底哪种传参方式更好一些呢?选择传址传参会更好一些,为什么呢?函数传参的时候,参数是需要压栈的,这样就会有时间和空间上的系统开销;如果传递一个结构体对象的时候,结构体过大,参数压栈的系统开销比较大,会导致性能的下降。所以结构体传参时,要传结构体的地址

3. 位段

位段的声明与结构体是类似的,但是存在着两个不同:

1. 位段的成员类型可以是 int , unsigned int 或 signed int ,char ,在C99中位段成员的类型也可以选择其它的类型
2. 位段的成员后面又一个冒号和数字

具体的声明如下代码所示:

struct Digital
{int m_a : 2;int m_b : 5;int m_c : 10;int m_d : 30;
};

Digital 就是一个位段。位段中的成员的冒号后面的数字是该成员变量的大小,单位是 bit ,该数字的大小不能超过成员的类型的大小,如成员的类型为 int 类型,那么该成员冒号后面的数字不能超过32。那么这个位段的大小是多少呢?又该怎么计算呢?哎!想要知道这些,就得要了解位段的内存分配:

1. 位段的成员可以是 int , unsigned int 或 signed int 或 char 等类型
2. 位段的空间是按照需要以 4个字节(int) 或者 1个字节(char) 的方式来开辟
3. 位段涉及很多不确定的因素,位段是不跨平台的,注重可移植的程序应当避免使用位段

根据位段的内存分配方式,来计算下面位段的大小:

struct Number
{char a : 3;char b : 4;char c : 5;char d : 4;
};int main()
{//将num对象中的所有比特位都赋值为0struct Number num = { 0 };num.a = 10;num.b = 12;num.c = 3;num.d = 4;return 0;
}

大概的来推算一下,该位段的大小。前面有提到过,位段的空间是是按照4或1个字节的方式开辟的,这个位段的成员类型大多是char类型,所以开辟空间的方式每次开辟1个字节,不够了再开辟1个字节。
首先为变量a开辟一个字节大小的空间,变量只有3bit大小,剩下5bit未被使用;接着存储变量b,变量的大小有4bit大小,而之前为变量a开辟的剩下的空间为5bit,所以不需要再次开辟空间,剩下的5bit空间足够存储变量b,这样就剩下一个bit大小的空间;
接着存储变量c,变量c有5bit大小,剩下的空间只有1bit大小,所以需要再次开辟1个字节大小的空间,第一次开辟的空间的剩余部分被舍弃,那么变量c就直接占用新开辟的空间的5bit大小空间,剩下3bit未被使用;
接下来再来存储变量d,变量d有4bit大小,但是剩下的空间只有3bit大小,所以需要再次开辟1个字节大小的空间,那么剩下的3bit大小的空间按照之前的假设,直接舍弃。
所以按照上面的推算位段Number的大小为3个字节。

但是还有另外一种推算方法就是剩下的空间不被舍弃,那么按照此推算方法,位段Number的大小为2个字节。

在当前环境下,结果到底是2个字节还是3个字节,用 sizeof 一求便知,结果如下:

在这里插入图片描述
根据结果易知,在vs编译器下推算1的方法正确。

前面有提到位段是不支持跨平台的,由于它有许多的不确定的因素,那么这些不确定的因素有:

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

既然位段有这么多的缺陷,那么它还有什么运用场景呢?位段可以运用于网络协议中,这里使用位段不仅能够实现想要的效果,还能节省空间,这样网络传输的数据包大小也会较小一些,对于网络的通畅有帮助。

在使用位段时,还需要注意:位段的几个成员共有同一个字节,这样有些成员的起始位置并不是某个字节的起始位置,那么这些位置处是没有地址的。内存中每个字节分配一个地址,而一个字节的内部的bit位是没有地址的。所以不能对位段的成员使用&操作符,如此一来就不能通过
scanf 直接给位段的成员输入值,只能是先输入一个值,并将该值放在一个变量中,然后赋值给位段的成员。

具体如下所示:

struct Digital dig = { 0 };
//错误方法 —— scanf("%d", &dig.m_a);//正确方法
int a = 0;
scanf("%d", &a);
dig.m_a = a;

4. 联合体

如同结构体一致,联合体也是由一个或者多个成员构成,这些成员可以是不同的类型,但是编译器只为最大的成员分配足够的内存空间。联合体也有它的关键字 —— union。联合体的特点是所有的成员都共用同一块内存空间,所以联合体也称为:共用体。给联合体当中的某个成员赋值,其它的成员也会跟着变化。下面来使用联合体:
在这里插入图片描述
如何计算联合体的大小?

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

接下来用具体的例子来明白如何计算联合体的大小:

在这里插入图片描述
联合体中也存在内存对齐,该联合体的最大对齐数为4,而该联合体中最大成员的大小为5,不是最大对齐数的整数倍,浪费3个字节大小的空间,所以 Un1 联合体的大小为8个字节。

下面使用联合体来判断当前机器是大端还是小端:首先来介绍何为大端和小端,如下图所示:

在这里插入图片描述
下面开始编写代码:

union Un
{char c;int i;
};int main()
{union Un un;un.i = 1;if (un.c == 1){printf("小端\n");}else{printf("大端\n");}return 0;
}

利用了联合体的一个特点:所有的成员都共用同一块内存空间,给联合体当中的某个成员赋值,其它的成员也会跟着变化。具体的分析如下:

在这里插入图片描述

5. 枚举

枚举枚举,顾名思义就是一一列举,把可能的值一一列,在日常生活中,天数可以被一一列举,月份可以被一一列举等等,而这些数据就可以使用枚举。枚举就是用来表示那些取值可以被一一列举的类型。枚举的关键字是 —— enum。枚举类型的使用:

//枚举类型的声明
enum Day
{//枚举类型的可能取值//这些可能取值都是常量,并且它们都是有值的//默认从 0 开始,依次递增 1Mon,Tues,Wed,Thur,Fri,Sat,Sun
};int main()
{//枚举类型的定义enum Day day = Sun;//打印枚举类型Day的所有可能取值printf("%d\t", Mon);printf("%d\t", Tues);printf("%d\t", Wed);printf("%d\t", Thur);printf("%d\t", Fri);printf("%d\t", Sat);printf("%d\t", Sun);return 0}

打印结果:

在这里插入图片描述
枚举类型中的常量是可以在声明的同时赋初始值的,也就是给常量初始化,如下所示:

在这里插入图片描述

有人可能会有疑问,既然是定义常量,为什么要使用枚举这个类型呢?#define也可以定义常量呀?也就是说,枚举有什么优点?枚举常量的优点有很多:

1. 增加代码的可读性与可维护性
2. 和#define定义的标识符比较枚举有类型检查,更加严谨
3. 便于调试,预处理阶段会删除#define定义的符号
4. 使用方便,一次可以定义多个常量
5. 枚举常量是遵循作用域规则的,枚举声明在函数内,就只能在函数内使用

6. 结言

以上就是C语言中主要的自定义类型。其中最主要的是结构体类型,在学习C++时,遇到的类与结构体有着相似之处。学好它,有助于理解C++中的类类型。

相关文章:

C语言中的自定义类型 —— 结构体.位段.联合体和枚举

自定义类型 1. 前言2. 结构体2.1 结构体的声明2.2 结构体变量的定义和初始化2.3 结构体的特殊声明2.4 结构体的自引用2.5 结构体的内存对齐2.6 修改默认对齐数2.7 结构体传参 3. 位段4. 联合体5. 枚举6. 结言 1. 前言 在C语言中已经为用过户提供了内置类型,如&…...

【序列贪心】摆动序列 / 最长递增子序列 / 递增的三元子序列 / 最长连续递增序列

⭐️个人主页:小羊 ⭐️所属专栏:贪心算法 很荣幸您能阅读我的文章,诚请评论指点,欢迎欢迎 ~ 目录 摆动序列最长递增子序列递增的三元子序列最长连续递增序列 摆动序列 摆动序列 贪心策略:统计出所有的极大值和极小…...

从零开发一个B站视频数据统计Chrome插件

从零开发一个B站视频数据统计Chrome插件 前言 B站(哔哩哔哩)作为国内最大的弹幕视频网站之一,视频的播放量、点赞、投币、收藏等数据对于内容创作者和数据分析者来说非常重要。本文将带你一步步实现一个Chrome插件,自动统计并展…...

【Python实战】飞机大战

开发一个飞机大战游戏是Python学习的经典实战项目,尤其适合结合面向对象编程和游戏框架(如Pygame)进行实践。以下是游戏设计的核心考虑因素和模块划分建议: 一、游戏设计核心考虑因素 性能优化 Python游戏需注意帧率控制&#xff…...

WebAPI项目从Newtonsoft.Json迁移到System.Text.Json踩坑备忘

1.控制器层方法返回类型不能为元组 控制器层方法返回类型为元组时,序列化结果为空。 因为元组没有属性只有field,除非使用IncludeFields参数专门指定,否则使用System.Text.Json进行序列化时不会序列化field var options new JsonSerializ…...

人工智能助力工业制造:迈向智能制造的未来

在当今数字化转型的浪潮中,人工智能(AI)技术正逐渐成为推动工业制造领域变革的核心力量。智能制造作为工业 4.0 的重要组成部分,通过将 AI 技术与传统制造工艺深度融合,正在重塑整个生产流程,提高生产效率、…...

影楼精修-露齿笑算法解析

注意,为避免侵权,本文图片均为AIGC生成或网络公开数据; 像素蛋糕-露齿笑 在介绍本文之前,先说一下,其实露齿笑特效,并非像素蛋糕首创,早在几年前,face app就率先推出了这个效果&am…...

【iview】es6变量结构赋值(对象赋值)

变量的解构赋值 以iview的src/index.js中Vue.prototype.$IVIEW改造为例练习下怎么使用变量的解构赋值 原来的写法: const install function(Vue, opts {}) {if (install.installed) return;locale.use(opts.locale);locale.i18n(opts.i18n);Object.keys(iview).fo…...

在Windows系统中使用Docker发布镜像到镜像仓库

在Windows系统中使用Docker发布镜像到镜像仓库的步骤如下: 步骤 1:安装并配置Docker 安装Docker Desktop • 下载Docker Desktop for Windows并安装。 • 确保启用WSL 2或Hyper-V后端(根据系统版本选择)。 验证Docker运行状态 打…...

糖尿病筛查常识---秋浦四郎

糖尿病筛查可以早期发现糖尿病或糖尿病前期(血糖异常但未达到糖尿病标准),以利于及时干预,预防并发症。因为许多人患上糖尿病时没有明显症状,但已经开始对身体造成损害,有了明显糖尿病症状才检查发现糖尿病…...

CSS 预处理器 Sass

目录 Sass 一、Sass 是什么? 二、核心功能详解 1. 变量(Variables) 2. 嵌套(Nesting) 3. 混合宏(Mixins) 4. 继承(Inheritance) 5. 运算(Operations&…...

Mybatisplus:一些常用功能

自动驼峰 mybatis-plus:configuration:# 开启驼峰命名规则,默认true开启map-underscore-to-camel-case: true# 控制台日志打印,便于查看SQLlog-impl: org.apache.ibatis.logging.stdout.StdOutImpl TableName 作用:表名注解,标识…...

Golang WaitGroup 用法 源码阅读笔记

使用 sync.WaitGroup可以用来阻塞等待一组并发任务完成 下面是如何使用sync.WaitGroup的使用 最重要的就是不能并发调用Add()和Wait() var wg sync.WaitGroupfor ... {wg.Add(1) // 不能和wg.Wait()并发执行go func() {// 不能在启动的函数里面执行wg.Add(), 否则会panicde…...

第二章:一致性基础 A Primer on Memory Consistency and Cache Coherence - 2nd Edition

在本章中,我们将介绍足够多的缓存一致性知识,以便理解一致性模型是如何与缓存相互作用的。我们在 2.1 节首先给出在本入门教程中所考虑的系统模型。为了简化本章以及后续章节的阐述,我们选择了尽可能简单的系统模型,该模型足以说明…...

C++类_移动构造函数

std::move 的主要用途是在对象所有权转移时,触发移动构造函数或移动赋值运算符,避免不必要的深拷贝,提升性能。 移动构造函数 和 移动赋值运算符, std::move转换为右值,匹配到移动构造函数和移动赋值运算符。…...

Spring AI 实战:第一章、Spring AI入门之DeepSeek调用

引言:当Spring遇上AI,会擦出怎样的火花? 作为一名Java开发者,是否曾经眼红Python阵营那些花里胡哨的AI应用?是否在对接各种大模型API时,被五花八门的接口规范搞得头大?好消息是,Spr…...

fastapi+vue中的用户权限管理设计

数据库设计:RBAC数据模型 这是一个典型的基于SQLAlchemy的RBAC权限系统数据模型实现,各模型分工明确,共同构成完整的权限管理系统。 图解说明: 实体关系: 用户(USER)和角色(ROLE)通过 USER_ROLE 中间表实现多对多关系…...

Space Engineers 太空工程师 [DLC 解锁] [Steam] [Windows]

Space Engineers 太空工程师 [DLC 解锁] [Steam] [Windows] 需要有游戏正版基础本体,安装路径不能带有中文,或其它非常规拉丁字符; DLC 版本 至最新全部 DLC 后续可能无法及时更新文章,具体最新版本见下载文件说明 DLC 解锁列表&…...

随机变量数字特征

主要介绍一维随机变量期望和方差、二维随机变量期望和方差、以及协方差相关公式,及推导。 一维随机变量 以一个抛硬币的场景作为例子,如下: 抛掷两枚均匀硬币,如果两枚都是正面向上,则赢得2元,否则就输掉…...

C++总结01-类型相关

一、数据存储 1.程序数据段 • 静态(全局)数据区:全局变量、静态变量 • 堆内存:程序员手动分配、手动释放 • 栈内存:编译器自动分配、自动释放 • 常量区:编译时大小、值确定不可修改 2.程序代码段 •…...

【多线程】七、POSIX信号量 环形队列的生产者消费者模型

文章目录 Ⅰ. 信号量一、POSIX 信号量的概念二、POSIX 信号量的类型区别三、POSIX 信号量与 SystemV 信号量的区别Ⅱ. 线程信号量基本原理一、为什么要引入信号量❓二、PV 操作三、POSIX 信号量的实现原理四、CAS操作介绍Ⅲ. POSIX未命名信号量接口一、初始化无名信号量二、销毁…...

二维码批量识别—混乱多张二维码识别-物品分拣—-未来之窗-仙盟创梦IDE

仙盟模型 用途 精准分拣:快速准确识别物品上复杂或多个二维码,依据码中信息(如目的地、品类等)实现物品自动化分拣,提高分拣效率与准确性。库存管理:识别入库、出库物品二维码,更新库存数据&am…...

《TensorFlow 与 TensorFlow Lite:协同驱动 AI 应用全景》

《TensorFlow 与 TensorFlow Lite:协同驱动 AI 应用全景》 摘要 :在机器学习技术浪潮中,TensorFlow 与 TensorFlow Lite 作为 Google 技术栈的核心组件,分别占据云端训练与端侧部署的关键位置。本文将系统梳理二者架构特性、功能…...

Spring AI 实战:第三章、Spring AI结构化输出之告别杂乱无章

引言:当程序员遇上剧荒 “周末看什么?” 这个看似简单的问题,往往能让我们在各大影视平台间反复横跳半小时,最后无奈选择重刷《老友记》。本期让我们用技术解决这个"世纪难题":让大模型成为你的私人影视推荐…...

ros2 humble 控制真实机械臂(以lerobot为例)

基础版 0.确保串口访问权限 sudo chmod 666 /dev/ttyARM0 # 确保串口访问权限 1.下载 lerobot 驱动功能包 git clone https://gitee.com/kong-yue1/lerobot_devices.git 2.编写控制节点(完整代码) 主要功能是与 Feetech 电机总线进行通信&#…...

REINFORCE蒙特卡罗策略梯度算法详解:python从零实现

🧠 向所有学习者致敬! “学习不是装满一桶水,而是点燃一把火。” —— 叶芝 我的博客主页: https://lizheng.blog.csdn.net 🌐 欢迎点击加入AI人工智能社区! 🚀 让我们一起努力,共创…...

模拟SIP终端向Freeswitch注册用户

1、简介 使用go语言编写一个程序,模拟SIP-T58终端在Freeswitch上注册用户 2、思路 以客户端向服务端Freeswitch发起REGISTER请求,告知服务器当前的联系地址构造SIP REGISTER请求 创建UDP连接,连接到Freeswitch的5060端口发送初始的REGISTER请…...

Elasticsearch 中的索引模板:如何使用可组合模板

作者:来自 Elastic Kofi Bartlett 探索可组合模板以及如何创建它们。 更多阅读: Elasticsearch:可组合的 Index templates - 7.8 版本之后 想获得 Elastic 认证吗?查看下一期 Elasticsearch Engineer 培训的时间! El…...

一篇文章看懂时间同步服务

Linux 系统时间与时区管理 一、时间与时钟类型 时钟类型说明管理工具系统时钟由 Linux 内核维护的软件时钟,基于时区配置显示时间timedatectl硬件时钟 (RTC)主板上的物理时钟,通常以 UTC 或本地时间存储,用于系统启动时初始化时间hwclock …...

Mysql常用语句汇总

Mysql语句分类 DDL: 数据定义语言,用来定义数据库对象(数据库、表、字段)DML: 数据操作语言,用来对数据库表中的数据进行增删改DQL: 数据查询语言,用来查询数据库中表的记录DCL: 数据控制语言,用来创建数据…...

迭代器的思想和实现细节

1. 迭代器的本质 迭代器是一种行为类似指针的对象,它可能是指针(如 std::vector 的迭代器),也可能是封装了指针的类(如 std::list 的迭代器)。如果是指针那天然就可以用下面的运算,如果是类&am…...

[Vue]编程式导航

在 Vue 中&#xff0c;编程式导航是通过 JavaScript 代码&#xff08;而非 <router-link> 标签&#xff09;动态控制路由跳转的核心方式。这个方法依赖于 Vue Router 提供的 API&#xff0c;能更灵活地处理复杂场景&#xff08;如异步操作、条件跳转等&#xff09;。 一、…...

使用Node.js搭建https服务器

一、引言 https是是http的安全版本&#xff0c;在http的基础上通过传输加密和身份认证保证了传输过程中的安全性。可以认为&#xff1a;https http tls/ssl。本文讲述使用Node.js搭建https服务器的方法。 二、编译OpenSSL 按照《Openssl在Linux下编译/交叉编译》&#xff0…...

网络安全:sql注入练习靶场——sqli_labs安装保姆级教程

网络安全&#xff1a;sql注入练习靶场——sqli_labs安装保姆级教程 前言 sqli-labs靶场是一个开源的sql注入练习的综合靶场&#xff0c;包含大部分sql注入漏洞以及注入方式 网络安全学习者可以通过在sqli-labs靶场练习提升对sql注入的理解&#xff0c;以及学习各种绕过姿势。…...

rfsoc petalinux适配调试记录

1。安装虚拟机 2.设置共享文件夹 https://xinzhi.wenda.so.com/a/1668239544201149先设置文件夹路径 vmware 12 下安装 ubuntu 16.04 后&#xff0c;按往常的惯例安装 vmware-tools&#xff0c;安装时提示建议使用 open-vm-tools&#xff0c;于是放弃 vmware-tools 的安装&am…...

Windows下编译WebRTC源码

一、开发环境要求 准备一台64位的win10或win11&#xff08;我用的是win11&#xff09;电脑。最好是一台纯净的、没有安装过其它软件的Windows主机&#xff0c;避免已安装的软件和库对编译造成影响。 然后最好预留超过100G的硬盘空间。因为编译WebRTC时会产生大量的临时文件需…...

【vscode】.dart文件没有错误波浪线

解决方法&#xff1a; 新建一个文件夹&#xff0c;在vscode里打开这个文件夹 在这个文件夹里新建.dart文件后打开即可出现错误波浪线...

OpenharmonyOS+RK3568,【编译烧录】

文章目录 1. 摘要 ✨2. 代码下载 &#x1f4e9;3. 编译 &#x1f5a5;️4. 修改&适配 ✂️4.1 编译框架基本概念4.2 vendor & device 目录4.3 内核编译4.3.1 如何修改、适配自己的开发板&#xff1f; 4.4 修改外设驱动 5. 烧录&验证 &#x1f4cb;参考 1. 摘要 ✨ …...

从零开始理解 C++ 后端编程中的分布式系统

一、什么是“分布式”? 简单来说,分布式系统就是由“多个计算机(或服务器)”组成的一个大系统,它们通过网络协作完成某个任务,就像一个“团队合作”一样。 想象你开了一家餐馆,最初只有 一个厨房 和 一个服务员,所有订单都在这里处理。随着生意变好,你需要: 开分店…...

基于SpringBoot的篮球竞赛预约平台设计与实现

1.1 研究背景 科学技术日新月异的如今&#xff0c;计算机在生活各个领域都占有重要的作用&#xff0c;尤其在信息管理方面&#xff0c;在这样的大背景下&#xff0c;学习计算机知识不仅仅是为了掌握一种技能&#xff0c;更重要的是能够让它真正地使用到实践中去&#xff0c;以…...

具身系列——PPO算法实现CartPole游戏(强化学习)

完整代码参考&#xff1a; https://gitee.com/chencib/ailib/blob/master/rl/ppo_cartpole.py 执行结果&#xff1a; 部分训练得分&#xff1a; (sd) D:\Dev\traditional_nn\feiai\test\rl>python ppo_cartpole_v2_succeed.py Ep: 0 | Reward: 23.0 | Running: 2…...

小程序与快应用:中国移动互联网的渐进式革命——卓伊凡的技术演进观

小程序与快应用&#xff1a;中国移动互联网的渐进式革命——卓伊凡的技术演进观 在知乎看到很多&#xff1a;“懂王”发布的要把内行笑疯了的评论&#xff0c;卓伊凡必须怼一下&#xff0c;真印证那句话&#xff0c;无知者无畏 一、Web与小程序的技术本质差异 1.1 浏览器渲染…...

Socket 编程 UDP

Socket 编程 UDP UDP 网络编程V1 版本 - echo serverV2 版本 - DictServerV3 版本 - 简单聊天室 补充参考内容地址转换函数关于 inet_ntoa UDP 网络编程 声明&#xff1a;下面代码的验证都是用Windows作为客户端的&#xff0c;如果你有两台云服务器可以直接粘贴我在Linux下的客…...

Lua 基础 API与 辅助库函数 中关于创建的方法用法

目录 基础 API 函数1. lua_len(L, index)2. lua_load(L, reader, data, chunkname, mode)3. lua_newstate(allocator, ud)4. lua_newtable(L)5. lua_newthread(L)6. lua_newuserdata(L, size)7. lua_next(L, index) 辅助库函数&#xff08;luaL_*&#xff09;8. luaL_len(L, in…...

YOLOv11改进:利用RT-DETR主干网络PPHGNetV2助力轻量化目标检测

这里写自定义目录标题 YOLOv11改进&#xff1a;利用RT-DETR主干网络PPHGNetV2助力轻量化目标检测1. 介绍2. 引言3. 技术背景3.1 YOLOv11概述3.2 RT-DETR与PPHGNetV23.3 相关工作 4. 应用使用场景5. 详细代码实现5.1 环境准备5.2 PPHGNetV2主干网络实现5.3 YOLOv11与PPHGNetV2集…...

centos7.0无法安装php8.2/8.3

在centos安装php8.2报错 configure: error: *** A compiler with support for C17 language features is required. 配置过程检测到你的系统编译器不支持 C17 语言特性&#xff0c;而 PHP 8.2 的编译需要编译器支持 C17 sudo yum update -y sudo yum install centos-releas…...

工业传动核心部件深度剖析:丝杆升降机与气缸的技术特性及选型指南

在工业自动化技术飞速发展的当下&#xff0c;丝杆升降机与气缸作为关键的直线传动部件&#xff0c;广泛应用于各类机械设备中。对于工程师而言&#xff0c;深入了解它们的技术特性、优缺点及适用场景&#xff0c;是实现高效、精准设备设计的重要前提。本文将从技术原理出发&…...

flask 获取各种请求数据:GET form-data x-www-form-urlencoded JSON headers 上传文件

在 Flask 里&#xff0c;能使用多种方法获取不同类型的请求数据&#xff0c;下面详细介绍常见请求数据的获取方式。 获取查询字符串参数&#xff08;GET 请求&#xff09; 查询字符串参数一般在 URL 里&#xff0c;以 ?key1value1&key2value2 这种形式存在。可通过 requ…...

c++_2011 NOIP 普及组 (1)

P1307 [NOIP 2011 普及组] 数字反转 P1307 [NOIP 2011 普及组] 数字反转 - 洛谷 # P1307 [NOIP 2011 普及组] 数字反转 ## 题目描述 给定一个整数 $N$&#xff0c;请将该数各个位上数字反转得到一个新数。新数也应满足整数的常见形式&#xff0c;即除非给定的原数为零&…...

​​信息泄露:网站敏感文件泄漏的隐形危机与防御之道​

在网络安全领域&#xff0c;​​信息泄露​​常被称为“沉默的杀手”。攻击者无需复杂漏洞&#xff0c;仅通过网站无意暴露的敏感文件&#xff08;如源码备份、配置文件、版本控制记录&#xff09;&#xff0c;即可获取数据库密码、API密钥甚至服务器权限。本文将深入剖析信息泄…...