摘要:一结构体的声明与定义结构体的声明结构是一些值的集合,这些值称为成员变量。但是结构体变量的变量名并不是指向该结构体的地址,所以要使用取地址运算符才能获取其地址。因此,结构体传参的时候,要传结构体的地址。
C语言提供了不同的数据类型,比如说int、float、double、char等,不同的类型决定了一个变量在内存中应该占据的空间以及表现形式。
但是,当我们定义一个人的时候,人的不同属性就比较难用同一个数据类型来定义了,因为人的身高、年龄、体重等属性往往需要不同数据类型,在这个时候,我们便引入结构体这个概念。
结构是一些值的集合,这些值称为成员变量。结构的每个成员可以是不同类型的变量
当我们面对的事物有多个不同的数据类型的时候,我们就可以使用结构体来组织了。
比如说,一本书有书名、作者、售价、出版日期等等不同的数据类型,这时候我们可以创建结构体来包含书的不同数据类型。
而结构体声明是描述结构体组合的主要方法,语法格式为:
struct 结构体名称
{
结构体成员1;
结构体成员2;
结构体成员3;
…
};//分号不能丢
【注意】
结构体成员既可以是任何一种基本的数据类型,也可以是另一种结构体,如果是后者就相当于结构体的嵌套。(俗称套娃)
例如:
struct Book//描述一本书的相关属性,其中Book是这个框架的名称{ char name[20];//书名 char author[20];//作者 float price;//价格};//分号一定不能丢
这样就相当于描述了一本书的框架。
结构成员的类型可以是标量、数组、指针、甚至是其他结构体。
结构体的声明只是进行一个简单的描述,实际上在没有定义结构体类型变量之前,它是不会在内存中分配空间的。
也就是说,它还没有被真正使用,虚拟存在,只有定义了结构体类型变量,才真实存在。
举个例子,上面定义了书的框架
struct Book//描述一本书的相关属性,其中Book是这个框架的名称{ char name[20];//书名 char author[20];//作者 float price;//价格};//分号一定不能丢
这里在编译器中,并不会分配内存空间,它仅仅是虚拟存在。而一旦我们定义了结构体类型变量,它就可以被分配空间了。
比如:
struct Book//描述一本书的相关属性,其中Book是这个框架的名称{ char name[20];//书名 char author[20];//作者 float price;//价格};//分号一定不能丢int main(){ struct Book book;//局部变量--放在栈区 return 0;}
我们在上面例子中也可以注意到,定义结构体变量的语法是:
struct 结构体名称 结构体变量名
此外,还可以在结构体声明的时候定义结构体变量
struct Book//描述一本书的相关属性{ char name[20]; char author[20]; float price;}b1,b2;//b1,b2是全局变量。放在静态区int main(){ struct Book book;//局部变量--放在栈区 return 0;}
b1、b2结构体变量是一个全局变量,在其他函数中也可以对它进行访问。
我们在定义一个变量或数组的时候可以对其进行初始化,
例如:
int a=10;int arr[10]={1,2,3,4,5,6,7,8,9,0};
同理,定义结构体变量的时候,我们也可以同时为其初始化
struct Book//描述一本书的相关属性{ char name[20]; char author[20]; float price;}b1,b2;//b1,b2是全局变量。放在静态区int main(){ struct Book book= { "《笑傲江湖》","金庸",30 };//这样的话,就将结构体变量初始化了,也就是定义变量的同时赋初值 return 0;}
结构体变量访问成员 结构变量的成员是通过点操作符(.)访问的。点操作符接受两个操作数。
比如,book.name就是引用book结构体变量的name成员,它是一个字符数组。
#include struct Book//描述一本书的相关属性{ char name[20]; char author[20]; float price;}b1, b2;//b1,b2是全局变量。放在静态区int main(){ struct Book book= { "《笑傲江湖》", "金庸", 30 };//这样的话,就将结构体变量初始化了,也就是定义变量的同时赋初值 printf("%s %s %f/n", book.name, book.author, book.price);//用. 来访问 return 0;}
如果访问嵌套的结构体成员的话,就需要使用多层点号运算符来进行操作。因为C语言的结构体只能对最底层的成员进行访问,如果存在多级结构体嵌套的话,就需要一级一级地深入,直到找到最底层的成员才行
struct S{ int a; char c; double d;};struct T{ struct S s;//结构体嵌套 char name[20]; int num;};int main(){ struct T t = { {100,"c",3.14},"里斯",30 }; printf("%d %c %f %s %d/n", t.s.a, t.s.c, t.s.d, t.name, t.num);//使用了两层点号运算符寻找成员 return 0;}
在开头的时候说过,结构的成员可以是标量、数组、指针。
在这里,我们来认识一下结构体指针。
struct Book *pt;
这里声明了一个指向Book结构体类型的指针变量pt
struct S{ int a; char c; double d;};struct T{ struct S s; char name[20]; int num;};int main(){ struct T t = { {100,"c",3.14},"里斯",30 }; printf("%d %c %f %s %d/n", t.s.a, t.s.c, t.s.d, t.name, t.num); struct T*pt = &t;//拿到地址的方式 printf("%d %c %f %s %d/n", (*pt).s.a, (*pt).s.c, (*pt).s.d, (*pt).name, (*pt).num); printf("%d %c %f %s %d/n",pt->s.a,pt->s.c,pt->s.d,pt->name,pt->num); return 0;}
【注意】数组名指向的是第一个元素的地址,所以可以直接将数组名赋值给指针变量。但是结构体变量的变量名并不是指向该结构体的地址,所以要使用取地址运算符(&)才能获取其地址。
如上面的:
struct T*pt = &t;//拿到地址的方式
通过上面的例子我们也可以发现,通过结构体指针访问结构体成员有以下两种方法:
(1)(*结构体指针).成员名
(2)结构体指针->成员名
第一种由于点号运算符(.)比指针的取值运算符(*)优先级高,所以要使用小口号先对指针进行解引用,让它变成该结构体变量,再用点运算符取访问其成员。
以上两种方法在实现的时候完全等价。但是,切记,点号(.)只能用于结构体,而箭头(->)只能用于结构体指针。
【打印结果一样】
当二者皆可用的时候,优先采用第二种方法,因为箭头具有指向性,很直观的就可以把它与指针联系起来了。
函数调用的时候,参数的传递就是值传递的过程,也就是将实参传给形参的过程。所以,结构体变量可以作为函数的参数传递,两个相同结构体类型的结构体变量也支持直接赋值。
struct S{ int arr[100]; int num; char ch; double d;};//结构体传参void print1(struct S ss){ printf("%d %d %d %c %1f", ss.arr[0],ss.arr[2],ss.num,ss.ch,ss.d);}//结构体地址传参void print2(struct S*ps){ printf("%d %d %d %c %1f", ps->arr[0], ps->arr[2], ps->num, ps->ch, ps->d);}int main(){ struct S s = { {1,2,3,4,5}, 100, "w",3.14 }; print1(s);//传结构体 print2(&s);//传地址 return 0;}
可以看到,确实把参数传递过去了。
那么,上面的 print1 和 print2 函数哪个好些?
答案是:首选print2函数。 原因:
函数传参的时候,参数是需要压栈的。 如果传递一个结构体对象的时候,结构体过大,参数压栈的的系统开销比较大,所以会导致性能的下降。
因此,结构体传参的时候,要传结构体的地址。
希望能对看到的大家有所帮助!
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/121141.html
摘要:为了减少在中创建的字符串的数量,字符串类维护了一个字符串常量池。但是当执行了方法后,将指向字符串常量池中的那个字符串常量。由于和都是字符串常量池中的字面量的引用,所以。究其原因,是因为常量池要保存的是已确定的字面量值。 String,是Java中除了基本数据类型以外,最为重要的一个类型了。很多人会认为他比较简单。但是和String有关的面试题有很多,下面我随便找两道面试题,看看你能不能...
摘要:结构体类型的特殊声明在初阶结构体中,我们已经将了结构体类型是如何进行声明的,那么在这里,我们将讲一些特殊的结构体声明不完全的声明。所以我们应该这样写通过指针来找到下一个同类型结构体的写法,我们就称之为结构体的自引用。 ...
摘要:摘要本文主要介绍了亚马逊的使用过程中发现的问题以及基于亚马逊实例自己搭建服务器的一些经验。之前公司使用亚马逊的实例,一切都非常好。但是我们架设在亚马逊实例上的服务器为了安全起见都是跨网段的,不支持,实现不了啊。 摘要 本文主要介绍了亚马逊RDS的使用过程中发现的问题以及基于亚马逊EC2实例自己搭建Mysql服务器的一些经验。 showImg(https://segmentfault.c...
阅读 1292·2023-04-26 03:05
阅读 721·2021-10-19 11:43
阅读 2975·2021-09-26 09:55
阅读 776·2019-08-30 15:56
阅读 943·2019-08-30 15:44
阅读 1171·2019-08-30 15:44
阅读 2666·2019-08-30 14:23
阅读 3191·2019-08-30 13:13