经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » C 语言 » 查看文章
一篇文章带你用C语言玩转结构体
来源:jb51  时间:2021/9/27 13:08:04  对本文有异议

前言

C语言提供了不同的数据类型,比如说int、float、double、char等,不同的类型决定了一个变量在内存中应该占据的空间以及表现形式。

但是,当我们定义一个人的时候,人的不同属性就比较难用同一个数据类型来定义了,因为人的身高、年龄、体重等属性往往需要不同数据类型,在这个时候,我们便引入结构体这个概念。

一、结构体的声明与定义

1.结构体的声明

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

当我们面对的事物有多个不同的数据类型的时候,我们就可以使用结构体来组织了。

比如说,一本书有书名、作者、售价、出版日期等等不同的数据类型,这时候我们可以创建结构体来包含书的不同数据类型。

而结构体声明是描述结构体组合的主要方法,语法格式为:

struct 结构体名称
{
结构体成员1;
结构体成员2;
结构体成员3;

};//分号不能丢

【注意】

结构体成员既可以是任何一种基本的数据类型,也可以是另一种结构体,如果是后者就相当于结构体的嵌套。(俗称套娃)

例如:

  1. struct Book//描述一本书的相关属性,其中Book是这个框架的名称
  2. {
  3. char name[20];//书名
  4. char author[20];//作者
  5. float price;//价格
  6. };//分号一定不能丢

这样就相当于描述了一本书的框架。

2.结构成员的类型

结构成员的类型可以是标量、数组、指针、甚至是其他结构体。

3.结构体的定义

结构体的声明只是进行一个简单的描述,实际上在没有定义结构体类型变量之前,它是不会在内存中分配空间的。

也就是说,它还没有被真正使用,虚拟存在,只有定义了结构体类型变量,才真实存在。

举个例子,上面定义了书的框架

  1. struct Book//描述一本书的相关属性,其中Book是这个框架的名称
  2. {
  3. char name[20];//书名
  4. char author[20];//作者
  5. float price;//价格
  6. };//分号一定不能丢

这里在编译器中,并不会分配内存空间,它仅仅是虚拟存在。而一旦我们定义了结构体类型变量,它就可以被分配空间了。

比如:

  1. struct Book//描述一本书的相关属性,其中Book是这个框架的名称
  2. {
  3. char name[20];//书名
  4. char author[20];//作者
  5. float price;//价格
  6. };//分号一定不能丢
  7. int main()
  8. {
  9. struct Book book;//局部变量--放在栈区
  10. return 0;
  11. }

我们在上面例子中也可以注意到,定义结构体变量的语法是:

struct 结构体名称 结构体变量名

此外,还可以在结构体声明的时候定义结构体变量

  1. struct Book//描述一本书的相关属性
  2. {
  3. char name[20];
  4. char author[20];
  5. float price;
  6. }b1,b2;//b1,b2是全局变量。放在静态区
  7. int main()
  8. {
  9. struct Book book;//局部变量--放在栈区
  10. return 0;
  11. }

b1、b2结构体变量是一个全局变量,在其他函数中也可以对它进行访问。

二、初始化结构体

我们在定义一个变量或数组的时候可以对其进行初始化,

例如:

  1. int a=10;
  2. int arr[10]={1,2,3,4,5,6,7,8,9,0};

同理,定义结构体变量的时候,我们也可以同时为其初始化

  1. struct Book//描述一本书的相关属性
  2. {
  3. char name[20];
  4. char author[20];
  5. float price;
  6. }b1,b2;//b1,b2是全局变量。放在静态区
  7. int main()
  8. {
  9. struct Book book=
  10. {
  11. "《笑傲江湖》","金庸",30
  12. };//这样的话,就将结构体变量初始化了,也就是定义变量的同时赋初值
  13. return 0;
  14. }

三、访问结构体成员

结构体变量访问成员 结构变量的成员是通过点操作符(.)访问的。点操作符接受两个操作数。

比如,book.name就是引用book结构体变量的name成员,它是一个字符数组。

  1. #include <stdio.h>
  2. struct Book//描述一本书的相关属性
  3. {
  4. char name[20];
  5. char author[20];
  6. float price;
  7. }b1, b2;//b1,b2是全局变量。放在静态区
  8. int main()
  9. {
  10. struct Book book=
  11. {
  12. "《笑傲江湖》", "金庸", 30
  13. };//这样的话,就将结构体变量初始化了,也就是定义变量的同时赋初值
  14. printf("%s %s %f\n", book.name, book.author, book.price);
  15. //用. 来访问
  16. return 0;
  17. }

在这里插入图片描述

四、结构体嵌套

如果访问嵌套的结构体成员的话,就需要使用多层点号运算符来进行操作。因为C语言的结构体只能对最底层的成员进行访问,如果存在多级结构体嵌套的话,就需要一级一级地深入,直到找到最底层的成员才行

  1. struct S
  2. {
  3. int a;
  4. char c;
  5. double d;
  6. };
  7. struct T
  8. {
  9. struct S s;//结构体嵌套
  10. char name[20];
  11. int num;
  12. };
  13. int main()
  14. {
  15. struct T t = { {100,'c',3.14},"里斯",30 };
  16. printf("%d %c %f %s %d\n", t.s.a, t.s.c, t.s.d, t.name, t.num);//使用了两层点号运算符寻找成员
  17. return 0;
  18. }

在这里插入图片描述

五、结构体指针

在开头的时候说过,结构的成员可以是标量、数组、指针。

在这里,我们来认识一下结构体指针。

struct Book *pt;

这里声明了一个指向Book结构体类型的指针变量pt

  1. struct S
  2. {
  3. int a;
  4. char c;
  5. double d;
  6. };
  7. struct T
  8. {
  9. struct S s;
  10. char name[20];
  11. int num;
  12. };
  13. int main()
  14. {
  15. struct T t = { {100,'c',3.14},"里斯",30 };
  16. printf("%d %c %f %s %d\n", t.s.a, t.s.c, t.s.d, t.name, t.num);
  17. struct T*pt = &t;//拿到地址的方式
  18. printf("%d %c %f %s %d\n", (*pt).s.a, (*pt).s.c, (*pt).s.d, (*pt).name, (*pt).num);
  19. printf("%d %c %f %s %d\n",pt->s.a,pt->s.c,pt->s.d,pt->name,pt->num);
  20. return 0;
  21. }

【注意】数组名指向的是第一个元素的地址,所以可以直接将数组名赋值给指针变量。但是结构体变量的变量名并不是指向该结构体的地址,所以要使用取地址运算符(&)才能获取其地址。

如上面的:

struct T*pt = &t;//拿到地址的方式

通过上面的例子我们也可以发现,通过结构体指针访问结构体成员有以下两种方法:

(1)(*结构体指针).成员名
(2)结构体指针->成员名

第一种由于点号运算符(.)比指针的取值运算符(*)优先级高,所以要使用小口号先对指针进行解引用,让它变成该结构体变量,再用点运算符取访问其成员。

以上两种方法在实现的时候完全等价。但是,切记,点号(.)只能用于结构体,而箭头(->)只能用于结构体指针。

【打印结果一样】

在这里插入图片描述

当二者皆可用的时候,优先采用第二种方法,因为箭头具有指向性,很直观的就可以把它与指针联系起来了。

六、结构体传参

函数调用的时候,参数的传递就是值传递的过程,也就是将实参传给形参的过程。所以,结构体变量可以作为函数的参数传递,两个相同结构体类型的结构体变量也支持直接赋值。

  1. struct S
  2. {
  3. int arr[100];
  4. int num;
  5. char ch;
  6. double d;
  7. };
  8. //结构体传参
  9. void print1(struct S ss)
  10. {
  11. printf("%d %d %d %c %1f", ss.arr[0],ss.arr[2],ss.num,ss.ch,ss.d);
  12. }
  13. //结构体地址传参
  14. void print2(struct S*ps)
  15. {
  16. printf("%d %d %d %c %1f", ps->arr[0], ps->arr[2], ps->num, ps->ch, ps->d);
  17. }
  18. int main()
  19. {
  20. struct S s = { {1,2,3,4,5}, 100, 'w',3.14 };
  21. print1(s);//传结构体
  22. print2(&s);//传地址
  23. return 0;
  24. }

在这里插入图片描述

可以看到,确实把参数传递过去了。

那么,上面的 print1 和 print2 函数哪个好些?

答案是:首选print2函数。 原因:

函数传参的时候,参数是需要压栈的。 如果传递一个结构体对象的时候,结构体过大,参数压栈的的系统开销比较大,所以会导致性能的下降。

因此,结构体传参的时候,要传结构体的地址。

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注w3xue的更多内容!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号