【前文】
首先我们需要先搞清楚两个问题。结构体是什么?结构体有什么用呢?关于第一个问题,结构体是可以存储一个或者多个类型成员变量的集合关于第二个问题,结构体作用,可以更加灵活地处理、描述一个复杂的属性。比如:一个学生的属性就是名字,年纪之类的,那么我们就需要用到结构体去描述
类型分类
类型主要分为内置类型和自定义类型,内置类型由 C++ 语言标准指定的基础类型,比如char,int就是 属于内置类型,而联合体,枚举,包括今天学习的结构体也是属于自定义类型
【正文】
一、结构体的声明
struct 自定义名称 { int _age; char _name; };
【注意】:
- 自定义名称是用于定义结构体类型,结构体内部只负责声明
- 定义结构体类型关键字是struct
- 结构体后面需要添加;
- 这里_可以更好表示这是一个结构体成员(不写也是可以的),提高可读性
1.1 特殊声明(匿名结构体)
struct//匿名结构体 { int _age; char _name; }*s; struct { int _age; char _name; }p;
【问题】:s=&p这里一段代码合法吗?
【回答】:
- 编译器会将这里两个变量看作两个不同类型的变量
- 基本只能使用一次匿名结构体,不然会区分不开
二、自引用
【问题一】:如果结构体里面内嵌结构体?
【回答】:
- 如果将结构体看成一个容器,如果将一个等大容器放进去,那么容器是不是被挤满了,再分析下,这样子结构体变量大小会无穷大!
【问题二】:如果想要结构体内嵌结构体,需要怎么做?
【回答】:
- 既然这个容器不能满足需求,可以找其他容器来保存数据,同时将它们之间创建联系—>将容器一半装入数据,一半是另外容器的指针,通过指针找到另外的容器
struct Su { int _a; struct Su *_p; };
三、typedef类型重命名相关的知识点
typedef struct { node *_p; }node;
【问题】:
特殊情况,如果是匿名结构体,node对类型的重命名,但是不能在结构体内提前使用,这里node还没有被定义
【解决措施】:
使用typedef就不要使用匿名结构体,而是直接以node作为结构体名称
四、结构体变量的创建和初始化(重点)
4.1 结构体变量的创建
struct Su { int _a; char _i; }p1; //声明类型的同时定义变量p1 struct Su P2;//定义变量p2 int main() { struct Su P3;//定义变量p3 return 0; }
【注意】:
- 这里p1,p2属于函数外的,称为全局变量
- 这里p3是在函数里面的,成为局部变量
【问题】:
struct node { int _val; }; int main { struct node * p; int ret = p -> _val; return 0; }
这里需要注意的是,如果只声明了结构体指针,是没有为结构体分配内存空间,这个指针指向的位置是不确定的,就是产生了野指针。对此在需要访问结构体成员变量时,通过定义结构体变量完成实例化,为结构体分配空间。
4.2 结构体变量的初始化
在定义结构体变量时,允许完成初始化,但是初始化一般需要讲究按照成员顺序初始化
struct { int _age; char _name; }s={18,"xiaoming"};
4.3 特殊初始化
特殊初始化主要分为两种:结构体嵌套初始化和不按照顺序初始化
结构体嵌套初始化 struct P { int _x; int _y; } struct Su { int _a; struct P _p; struct Su * _s; }struct Su n1={18, {1, 8}, NULL}; struct Su n2={20, {1, 7}, NULL};//结构体嵌套初始化 不按照顺序初始化 struct Su { int _age; char _name; }; int main() { struct Su s={.name="xiaoming", .age=18};//这样子不按照顺序也能完成初始化 return 0; }
五、结构成员访问操作符
访问结构体成员变量,我们可以通过以下两种方式实现
- 结构体变量.成员变量名
- 结构体指针->成员变量名
struct Su { int _a; char _i; }s; void Transmit1(struct Su s) { printf("%d",s.a);-->4 } void Transmit2(struct Su* s) { printf("%d",s->a);-->4 } int main() { Transmit1(s);//传结构体 Transmit2(&s);//传地址 return 0; }
【问题】:
在结构体传参中,推荐使用Transmit1和Transmit2哪个函数?
【回答】:
推荐使用Transmit2。函数传参中,参数会压栈,会空间和时间上的系统开销。如果当传参一个结构体对象的大小很大,那么参数压栈的系统开销比较大,导致性能下降