用户自己建立自己的结构体类型

1、  定义和使用结构体变量

(1)、结构体的定义

C语言允许用户自己建立由不同类型数据组成的组合型的数据结构,它称为结构体

(2)、声明一个结构体类型的一般形式为:

Struct 结构体名

{ 成员列表 };

1、  定义结构体类型变量

前面的也只是搭建了一个结构体类型,它相当于一个模型,并没有定义变量,其中并无具体数据,系统对之也不分配存储单元。

可以采取以下三种方法定义结构体类型的变量:

(1)、先声明结构体类型,再定义该类型的变量

(2)、在声明类型的同时定义变量

上面这种定义的一般形式为:

(3)、不指定类型名而直接定义结构体类型变量

注意:

1、  结构体类型和结构体变量是不同的概念,不要混同。区别:只能对变量赋值、存取或运算,而不能对一个类型赋值、存取或运算;在编译时,对类型是不分配空间的,只对变量分配空间。

2、  结构体类型中的成员名可以与程序中的变量名相同,但二者不代表同一对象。

3、  对结构体变量中的成员可以单独使用,它的作用与地位相当于普通变量。

1、  结构体变量的初始化与引用

在定义结构体变量时,可以对他进行初始化,即赋予初始值。

(1)、在定义结构体变量时可以对它的成员进行初始化。初始化列表是用花括号括起来的一些常量,这些常量依次赋给结构体变量中的各成员。注意:对结构体变量初始化,而不是对结构体类型初始化。

(2)、可以引用结构体变量中成员的值,引用方式为:结构体变量名.成员名

“.”是成员运算符。它在所有的运算符中优先级最高。因此可以把student1.name作为一个整体来看待,相当于一个变量。

注意:不能企图输出结构体变量名来达到输出结构体变量所有成员的值。只能对结构体变量中的各个成员分别进行输入和输出。

(3)、如果成员本身又属于一个结构体类型,则要用若干个成员运算符,一级一级的找到最低的一级的成员。只能对最低级的成员进行赋值、存取或运算。

(4)、对结构体变量的成员可以像普通变量一样进行各种运算。

由于“.”运算符的优先级最高,因此student1.age++是对(student1.age)进行自加运算,而不是先对age进行自加运算。

(5)、同类的结构体变量可以互相赋值。

说明:结构体变量的地址主要用作函数参数,传递结构体变量的地址。

==========================================================================

使用结构体数组

1、  定义结构体数组

(1)、定义结构体数组的一般形式是:

a、

b.先声明结构体类型(如struct Person),然后再用此类型定义结构体数组:

(2)、对结构体数组初始化的形式是在定义数组的后面加上:

==================================================================================

结构体指针

所谓结构体指针就是指向结构体变量的指针,一个结构体变量的起始地址就是这个结构体变量的指针。

1、  指向结构体变量的指针

指向结构体变量的指针变量既可以指向结构体变量,也可以指向结构体数组中的元素。指针变量的基类型必须与结构体变量的类型相同。

说明:为了使用方便和直观,C语言允许把(*p).name用p->name来代替,“->”代表一个箭头,p->name表示p指向结构体变量中的name成员。同样,

(*p).name等价于p->name。“->”称为指向运算符。

如果p指向一个结构体变量stu,以下三种用法等价:

(1)、stu.成员名(如:stu.name);

(2)、(*p).成员名(如:(*p).name);

(3)、p->成员名(p->name)。

2、指向结构体数组的指针

可以用指针变量指向结构体数组的元素。

注意:

如果p的初值为stu,即指向stu的第一个元素,p加1后,p就指向下一个元素,

例如:

请注意以上2点的不同。

(2)、程序定义了p,是一个指向struct Student类型对象的指针变量,它用来指向一个struct Student类型的对象,不应用来指向stu数组元素中的某一成员。例如下面的用法是不对的:

编译时将给出“警告”信息,表示地址的类型不匹配,不要认为反正p是存放地址的,可以将任何地址赋给它。如果要将某一成员的地址赋给p,可以用强制类型转换。先将成员的地址转换成p的类型。例如:

此时p的值是stu[1]元素的name成员的起始地址。可以用“   ”输出stu[1]中成员name的值。但是,p仍保持原来的类型,如果执行“ ”,则会输出stu[2]中name的值。执行p++时,p的值增加了结构体struct Student的长度。

3、用结构体变量和结构体变量的指针作函数参数

将一个结构体变量的值传递给另一个函数,有3中方法:

(1)、用结构体变量的成员作参数。例如:用stu[1].name或stu[2].name作为函数实参,将实参的值传递给形参。用法和用普通变量作实参是一样的,属于“值传递方式”。应当注意实参与形参的类型保持一致。

(2)、用结构体变量作实参。用结构体变量作实参时,采取的也是“值传递”的方式,将结构体变量所占的内存单元的内容全部按顺序传递给形参,形参也必须是同类型的结构体变量,在函数调用期间形参也要占用内存单元,这种传递方式在空间和时间上的开销非常大,如果结构体的规模很大时,开销是很可观的。此外,由于采用值传递方式,如果在执行被调用函数期间改变了形参(也是结构体变量)的值,该值不能返回主调函数,这往往造成使用上的不便。

(3)、用指向结构体变量(或数组元素)的指针作实参,将结构体变量(或数组元素)的地址传给形参。

C语言中的结构体的更多相关文章

  1. C语言中的结构体,结构体数组

    C语言中的结构体是一个小难点,下面我们详细来讲一下:至于什么是结构体,结构体为什么会产生,我就不说了,原因很简单,但是要注意到是结构体也是连续存储的,但要注意的是结构体里面类型各异,所以必然会产生内存 ...

  2. C语言中处理结构体的原理

    汇编中有几种寻址方式,分别是直接寻址:(ds:[idata]).寄存器间接寻址(ds:[bx]).寄存器相对寻址(ds:[bx + idata].ds:[bx + si])基址变址寻址(ds:[bx ...

  3. Verilog缺少一个复合数据类型,如C语言中的结构体

    https://mp.weixin.qq.com/s/_9UsgUQv-MfLe8nS938cfQ Verilog中的数据类型(Data Type)是分散的,缺少一个复合数据类型:把多个wire, r ...

  4. C语言中的结构体和C++中的结构体以及C++中类的区别

    c++中结构体可以定义一个函数 C中的结构体和C++中结构体的不同之处:在C中的结构体只能自定义数据类型,结构体中不允许有函数,而C++中的结构体可以加入成员函数. C++中的结构体和类的异同: 一. ...

  5. c语言中的结构体指针类型的cast

    1.我们在c语言中会经常碰到强制类型转换. 在这,我介绍一种结构pointer类型转换,但是有前提(有点类似于c++中的继承中的子父对象的cast). 简单的介绍一下: 首先我们要知道一个结构的指针, ...

  6. 018_go语言中的结构体

    代码演示 package main import "fmt" type person struct { name string age int } func main() { fm ...

  7. Go语言中的结构体 (struct)

    Golang官方称Go语言的语法相对Java语言而言要简洁很多,但是简洁背后也灵活了很多,所以很多看似很简单的代码上的细节稍不注意就会产生坑.本文主要对struct结构体的相关的语法进行总结和说明. ...

  8. C语言中全局结构体指针隐含的错误

    前天在嵌入式系统上,调试一个数组的全局变量时,发现该变量一直会动态变化.深入分析, 才发现该全局结构体没有申请内存,而是用了一个指针.这种情况编译器是检查不出来的,在linux 上运行会挂掉,但是在裸 ...

  9. C语言中的结构体是怎么定义的_怎么使用?

    结构体的定义 // 定义结构体st struct st{ int a; // 成员a int b; // 成员b }; #include <stdio.h> struct st{ int ...

随机推荐

  1. .NET Core的日志[4]:将日志写入EventLog

    面向Windows的编程人员应该不会对Event Log感到陌生,以至于很多人提到日志,首先想到的就是EventLog.EventLog不仅仅记录了Windows系统自身针对各种事件的日志,我们的应用 ...

  2. C++的内存泄漏检测

    C++大量的手动分配.回收内存是存在风险的,也许一个函数中一小块内存泄漏被重复放大之后,最后掏空内存. 这里介绍一种在debug模式下测试内存泄漏的方法. 首先在文件的开头以确定的顺序写下这段代码: ...

  3. 使用cmake自动构建工程

    公司引擎是用cmake根据目标平台来构建工程的,刚接触的时候深深体会到cmake的方便:如果目标平台是windows,它可以帮你自动构建出vs工程:如果是安卓,自动构建出eclipse工程,如果是IO ...

  4. Redis百亿级Key存储方案(转)

    1 需求背景 该应用场景为DMP缓存存储需求,DMP需要管理非常多的第三方id数据,其中包括各媒体cookie与自身cookie(以下统称supperid)的mapping关系,还包括了supperi ...

  5. Lind.DDD.Aspects通过Plugins实现方法的动态拦截~Lind里的AOP

    回到目录 .Net MVC之所以发展的如些之好,一个很重要原因就是它公开了一组AOP的过滤器,即使用这些过滤器可以方便的拦截controller里的action,并注入我们自己的代码逻辑,向全局的异常 ...

  6. OSGi规范的C#实现开源

    这是大约在3-4年前完成的一个C#实现的OSGi框架,实现的过程参照了OSGi规范与与一些实现思路(感谢当时的那些资料与项目),此框架虽然仅在几个小型项目有过实际的应用,但OSGi的规范实现还是相对比 ...

  7. javascript排序

    利用array中的sort()排序 w3cfunction sortNumber(a,b) { return a - b } var arr = new Array(6) arr[0] = " ...

  8. 【干货分享】流程DEMO-离职流程

    流程名: 离职申请   流程相关文件: 流程包.xml WebService业务服务.xml WebService.asmx WebService.cs   流程说明: 流程中集成了webservic ...

  9. mysql数据库主从同步

    环境: Mater:   CentOS7.1  5.5.52-MariaDB  192.168.108.133 Slave:   CentOS7.1  5.5.52-MariaDB  192.168. ...

  10. 支付宝AR抢红包?前端轻松就破解~

    近期阿里搞了各LBS+AR实景的红包玩法,小伙伴们在公司里都玩疯了~ 有时候为了抢一个红包,会跑到另一个地方去拍照,虽然略麻烦,但整体的互动还是很有意思的. 不过对于机智的前端童鞋来说,只需要简单的一 ...