文章目录
前言
今天夏目带你一起学习 结构体,枚举,联合,本篇博客除了结构体是我们以后要经常使用的外,枚举,联合却是比较少见的,但是就是因为少见我们就不学习了么?答案不是的,今天就带你学习这些知识。
一、结构体
1.1 结构的基础知识
1.2 结构的声明
描述一个学生:
struct Stu
{
char name[20];//名字
int age;//年龄
char sex[5];//性别
char id[20];//学号
}; //分号不能丢
1.3 特殊的声明
//匿名结构体类型
struct
{
i nt a;
char b;
float c;
}x;
struct
{
int a;
char b;
float c;
}a[20], *p;
那么问题来了?
//在上面代码的基础上,下面的代码合法吗?
p = &x;
警告:
编译器会把上面的两个声明当成完全不同的两个类型。
所以是非法的。
1.4 结构的自引用
//代码1
struct Node
{
int data;
struct Node next;
};
//可行否?
如果可以,那sizeof(struct Node)是多少?
答案是不可以的,因为结构体无法形成树形结构,因为这样会无穷递归下去,因此正确的写法是:
struct Node
{
int data;
struct Node* next;//写成指针形式
};
typedef struct Node
{
int data;
struct Node* next;
}Node; //常用,因为方便更改名字和类型
这里后续的数据机构会继续深入学习结构体,这里要做到了解和理解。
1.5 结构体变量的定义和初始化
有了结构体类型,那如何定义变量,其实很简单,下面就来给大家讲解:
定义方式一:
#include<stdio.h>
struct point
{
int x;
int y;
}p1; //声明类型的同时定义变量p1
定义方式二:
struct Point p2; //定义结构体变量p2
定义方式三:
typedef struct Point
{
int data;
struct Node* next;
}p3; //常用,因为方便更改名字和类型
初始化方式一:
struct Point p4 = {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.6 结构体内存对齐
学习之前,我想给大家看一张基础知识图片,这样有了基础再理解会事半功倍
- 结构体的第一个成员,对齐到结构体在内存中存放位置的0偏移处。
- 从第二个成员开始,每一个成员都要对齐到(一个对齐数)的整数倍处
- 对齐数:结构体成员自身大小和默认对齐数的较小值
- 结构体的总大小,必须是所有成员的对齐数中最大对齐数的整数倍
- 如果结构体中嵌套了结构体成员,要将嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处
- 结构体的总大小必须是最大对齐数的整数倍,这里的最大对齐数:包括嵌套结构体成员中的对齐数,的所有对齐数中的最大值。
下面就配合例题来讲解知识:
在32位系统环境,编译选项为4字节对齐,那么sizeof(A)和sizeof(B)是( )
struct A
{
int a;
short b;
int c;
char d;
};
struct B
{
int a;
short b;
char c;
int d;
}
所以答案是:16,12;
为什么存在内存对齐?
平台原因(移植原因)
性能原因:
总体来说
1.7 修改默认对齐数
#include <stdio.h>
#pragma pack(8)//设置默认对齐数为8
struct S1
{
char c1;
int i;
char c2;
};
#pragma pack()//取消设置的默认对齐数,还原为默认
#pragma pack(1)//设置默认对齐数为1
struct S2
{
char c1;
int i;
char c2;
};
#pragma pack()//取消设置的默认对齐数,还原为默认
int main()
{
//输出的结果是什么?
printf("%d\n", sizeof(struct S1));
printf("%d\n", sizeof(struct S2));
return 0;
}
1.8 结构体传参
struct S
{
int data[1000];
int num;
};
struct S s = {{1,2,3,4}, 1000};
//结构体传参
void print1(struct S s)
{
printf("%d\n", s.num);
}
//结构体地址传参
void print2(struct S* ps)
{
printf("%d\n", ps->num);
}
int main()
{
print1(s); //传结构体
print2(&s); //传地址
return 0;
}
总结
今天夏目带你一起学习 结构体,枚举,联合,本篇博客除了结构体是我们以后要经常使用的外,枚举,联合却是比较少见的