邀请加入嵌入式社区,您可以在上面发布问题,博客链接,公众号分享,行业消息,招聘信息等。
目录
结构体有什么用?
结构体声明
正常的结构体声明
匿名结构体
匿名结构体只有在创建的时候可以建立变量
两个相同的匿名结构体,编译器会认为是不相同
结构体自引用
结构体自引用结构体
结构体自引用结构体指针
typedef与结构体
只创建一个名称
创建两个变量名
注意事项
赋值
引用数据
结构体有什么用?
我们学习结构体之前需要知道结构体有什么用。举个例子,假如我现在需要将100个学生的信息录入教学系统。他们有年龄,性别,名字。我们进行编程的时候。不可能一一进行变量创建。那样太麻烦了。
那么我们就可以创建一个结构体,这个结构体类型里面包含年龄,性别,名字。这样,我们可以把小明,小红,小华都设置为这个类型的结构体。然后再对这个结构体赋值就可以了。
结构体声明
正常的结构体声明
结构体声明有两种方法
第一种是struct Stu s2。需要注意,struct一定要写上!
第二种方法就是在声明结构体的时候,}后面加上需要设置的结构体变量。
//现在我们建立了一个结构体类型,这个结构体类型名字叫做Stu
//s3,s4,s5,s6为全局变量,s1,s2是局部变量,但是他们都是同一种类型的结构体变量struct Stu
{char name[20];//名字char tele[12];//电话char sex[10];//性别int age;
} s4,s5,s6; //注意,这里的;不能少struct Stu s3;//全局变量int main()
{//创建的结构体变量struct Stu s1;//注意,这里的struct不能少struct Stu s2;return 0;
}
匿名结构体
匿名结构体只有在创建的时候可以建立变量
匿名结构体就是在创建结构体的时候,struct后面不加名字。个人不推荐使用匿名结构体,因为你创建了结构体之后,就不能再次创建该结构体的变量了。
以下面这个为例子,我们struct后面没有增加名字。那么现在就只有sa这一个变量是属于该结构体类型变量。不能再次建立该类型的结构体变量了。
struct
{int a;char c;
}sa; //这里一定要加sa,否则你接下来不能再次建立该结构体类型的变量了int main()
{struct s1; //这样是不可以的return 0;
}
两个相同的匿名结构体,编译器会认为是不相同
我们都知道,char型指针如果指向一个int型数据。编译器会报警告。
int main()
{int x;char* y = &x; //警告return 0;
}
匿名结构体依旧如此,我们看起来长得一样的两个匿名结构体,编译器也会将他们看成是两个不同的类型。
struct
{int a;char c;
}sa;struct
{int a;char c;
}* psa;int main()
{psa = &sa; //警告,编译器会把上面的两个声明当成完全不同的两个类型return 0;
}
结构体自引用
结构体自引用结构体
结构体创建变量的时候,创建的变量中有一个是自己,类似于递归。(但是不同!)
struct Node
{int data;//4struct Node n; //这样是错误的
};
上面这一串代码明显是错误的。因为,假如他是正确的,那么他的大小sizeof(struct Node)是多少呢?很显然,无法确认。
结构体自引用结构体指针
(1)数据结构中有一个东西叫做链表。
(2)因为我们平常使用的数据都是依次按照顺序连接的,但是,如果我现在需要在这一堆数据arr的中间假如一个其他数据a。那么现在我就需要将arr中间的数据改为a,然后更改的数据都往后移动。这样明显很麻烦。
(3)那么我们可以利用链表的方式,比如说,现在我们有一个数据链arr1。每一个数据指向后面这个数字。那么,我现在需要在中间增加一个数据b,只需要将插入前面这个数据指向b,然后b指向插入后面这个数据即可。
(4)那么,方法如下
struct Node
{int data;struct Node* next;
};
此时sizeof(struct Node)为多少呢?如果是32位的系统下,是8个字节。因为int是4个字节,后面这个指针在32位系统下也是4个字节。
typedef与结构体
只创建一个名称
因为我们每次创建一个变量需要写成 struct Stu s1;很明显太麻烦了。于是我们可以使用typedef与结构体给结构体起一个别名。
typedef struct Node
{double d;int data;
}Node;int main()
{//下面这两个等价struct Node n1;Node n2;return 0;
}
创建两个变量名
有时候我们会看见typedef与结构体之后会有两个名字,这个是怎么会是呢?
这样就同时建立了两种类型,一种是Node型的结构体,一个是Node型的结构体指针。
// 在重命名结构体时,命名了两种:Node结构体类型,*pNode结构体指针类型
typedef struct Node
{double d;int data;
}Node,*pNode;int main()
{pNode = &Node;return 0;
}
注意事项
虽然我们有typedef,但是如果需要链表那种形式的创建,还是有讲究的地方。我们看下面这一段代码。
因为在我typedef之前,你就已经使用了Node了。那么编译器就不会认识Node*。
/***** 错误写法 *****/
typedef struct
{double d;int data;//4Node* next;//4/8
}Node;/***** 正确写法 *****/typedef struct Node
{double d;int data;//4struct Node* next;//4/8
}Node;
赋值
对于结构体只能在定义的时候初始化才能全部赋值,之后就不能再全体赋值了,只能单个赋值。
/******这样是可以的,在定义变量的时候就初始化了*******/struct book s1={ "guojiajiaoyun", "yuwen",22.5};/******这种就不行了,在定义变量之后,若再要对变量的成员赋值,那么只能单个赋值了*******/struct book s1;//这样就是不行的,只能在定义的时候初始化才能全部赋值,之后就不能再全体赋值了,只能单个赋值s1={"guojiajiaoyun","yuwen",22.5 };s1.title = "yuwen";//只能单个赋值;
引用数据
我们直接写上'.'进行数据引用
struct T
{double weight;short age;
};struct S
{char c;struct T st;int a;double d;char arr[20];
};int main()
{struct S s = { 'c', {55.6, 30}, 100, 3.14, "hello bit" };printf("%c %d %lf %s\n", s.c, s.a, s.d, s.arr);printf("%lf\n", s.st.weight);return 0;
}