微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

【Struct(结构体)杂谈之二】名不正则言不顺---Struct(结构体)的声明、定义及初始化

Struct(结构体)的声明、定义及初始化

上一篇里我们讲了为什么我们要引入Struct这个数据类型,我们了解到Struct是一种聚合数据类型,是为了用户描述和解释一些事物的方便而提出的,Struct是一种用户自定义数据类型,如下图所示:




其实从理论上讲,数据类型就是人为制订的如何解释内存中的二进制数的协议,也就是说一个数字对应着一块内存(可能4字节,也可能20字节),而这个数字的类型则是附加信息,以告诉编译器当发现有对那块内存的操作语句(即某种操作符)时,要如何编写机器指令以实现那个操作。比如两个char类型的数字进行加法操作符操作,编译器编译出来的机器指令就和两个long类型的数字进行加法操作的不一样,也就是所谓的“如何解释内存中的二进制数的协议”。


具体到我们之前的例子来说,只是指定了一种结构体类型,它相当于一个模型,但其中并无具体数据,系统也不为之分配实际的内存单元。为了能在程序中使用结构体类型的数据,应当定义结构体类型的变量,并在其中存放具体的数据。


本篇将详细对Struct的声明、定义和初始化进行分析。


一、Struct的声明

要了解Struct的声明,我们需要首先了解声明的含义到底是什么?

---声明是要求编译器产生映射元素的语句。所谓的映射元素,就是前面介绍过的变量及函数,都只有3栏(或3个字段):类型栏、名字栏和地址栏(成员变量类型的这一栏就放偏移值)。即编译器每当看到声明语句,就生成一个映射元素,并且将对应的地址栏空着,然后留下一些信息以告诉连接器——此*.obj文件(编译器编译源文件生成文件)需要一些符号,将这些符号找到后再修改并完善此*.obj文件,最后链接


具体到上一回的例子,我们假如在另外一个文件中需要使用struct ExpectedBoyFriend,那么就需要在该源文件使用之前处使用下面的声明语句:

  1. externstructExpectedBoyFriend;

二、Struct的定义

上一小节我们了解了声明的定义,那么定义是什么呢?

---定义是要求编译器填充前面声明没有书写的地址栏。也就是说某变量对应的地址,只有在其定义时才知道。因此实际的在栈上分配内存等工作都是由变量的定义完成的,所以才有声明的变量并不分配内存。但应注意一个重点,定义是生成映射元素需要的地址,因此定义也就说明了它生成的是哪个映射元素的地址,而如果此时编译器的映射表(即之前说的编译器内部用于记录映射元素的变量表、函数表等)中没有那个映射元素,即还没有相应元素的声明出现过,那么编译器将报错。

在这里我们需要说下C和C++在定义Struct的区别, 先看下面2段代码

copy
    #include<iostream>
  1. usingnamespacestd;
  2. structSIMPLE
  3. {
  4. inta;
  5. charb;
  6. floatc;
  7. };
  8. SIMPLEx;

再看下面一段源码:
copy
    #include<stdio.h>
  1. structS0
  2. charmName[10];
  3. intmBornYear;
  4. };
  5. typedefstruct_S1
  6. }
  7. S1;
  8. S0sa;
  9. S1sb;

那么上面的代码中对Struct的定义都对了吗?
熟悉C/C++的同学应该能够马上知道下面的代码错了?
为什么呢?
因为 C语言中对于Struct的定义是需要使用struct S0 sa这种方式。

三、C99标准下的Struct的初始化方法

Struct的常见初始化方法我们可以在任何一本关于C语言书里面都可以找到,这里就不赘述了。我们先看下面一段代码

  1. staticstructusb_driverusb_storage_driver={
  2. .owner=THIS_MODULE,
  3. .name=\"usb-storage\",
  4. .probe=storage_probe,108); list-style:decimal-leading-zero outside; color:inherit; line-height:17.27272605895996px; margin:0px!important; padding:0px 3px 0px 10px!important"> .disconnect=storage_disconnect,
  5. .id_table=storage_usb_ids,};

我们在阅读GNU/Linux内核代码时,我们会经常遇到上述这样一种特殊的结构初始化方式,这种方式称为指定初始化。这种初始化方法源自C99标准。


以下文字摘录了C Primer Plus第五版中相关章节的内容,从而就可以很好的理解Linux内核采用这种方式的优势就在于由此初始化不必严格按照定义时的顺序。

已知一个结构,定义如下

copy
    structBook
  1. {
  2. chartitle[MAXTITLE];
  3. charauthor[MAXAUTHOR];
  4. floatvalue;
  5. };

C99支持结构的指定初始化项目,其语法与数组的指定初始化项目近似。只是,结构的指定初始化项目使用点运算符和成员名(而不是方括号和索引值)来标识具体的元素。例如,只初始化book结构的成员value,可以这样做:

copy
    structBooksurprise=
  1. .value=10.99
  2. };

可以按照任意的顺序使用指定初始化项目:

copy
    structBookgift=
  1. .value=25.99,248); line-height:17.27272605895996px; margin:0px!important; padding:0px 3px 0px 10px!important"> .author=\"Jamesbroadfool\",
  2. .title=\"Rueforthetoad\"
  3. };

正像数组一样,跟在一个指定初始化项目之后的常规初始化项目为跟在指定成员后的成员提供了初始值。另外,对特定成员的最后一次赋值是它实际获得的值。例如,考虑下列声明:

copy
    .value=18.90,248); line-height:17.27272605895996px; margin:0px!important; padding:0px 3px 0px 10px!important"> .author=\"hilionnapestle\",108); list-style:decimal-leading-zero outside; color:inherit; line-height:17.27272605895996px; margin:0px!important; padding:0px 3px 0px 10px!important"> 0.25
  1. };

这将把值0.25赋给成员value,因为它在结构声明中紧跟在author成员之后。新的值0.25代替了早先的赋值18.90。


出处:http://blog.csdn.net/tcpipstack/article/details/8267339

原文地址:https://www.jb51.cc/regex/360160.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐