C语言中结构体详解

2018-01-20

大家都知道学习嵌入式培训讲师东哥为大家带来关于:C语言中结构体详解,希望能帮助到更多的嵌入式学习者。

1 概述

C语言允许用户自己指定这样一种数据结构,它由不同类型的数据组合成一个整体,以便引用,这些组合在一个整体中的数据是互相联系的,这样的数据结构称为结构体,它相当于其它高级语言中记录。

声明一个结构休类型的一般形式如下:

struct 结构体名

{成员列表};

结构体名,用作结构体类型的标志,它又称 结构体标记,大括号内是该结构体中的各个成员,由它们组成一个结构体,对各成员都应进行类型声明如:

类型名 成员名;

也可以把 成员列表称为 域表,第一个成员也称为结构体中的一个域。成员名定名规则写变量名同。

struct student

{

int num;

char name[20];

char sex;

int age;

float score;

char addr[30];

};

2 定义结构体类型变量的方法

前面只是指定了一个结构体类型,它相当于一个模型,但其中并无具体数据,系统对之也不分配实际内存单元,为了能在程序中使用结构类型的数据,应当定义结构体类型的变量,并在其中存放具体的数据,可以采取以下3种方法定义结构体类型变量。

(1)先声明结构体类型再定义变量名

如上面已定义了一个结构体类型 struct student,可以用它来定义变量。如:

struct student  //结构体类型名

student1, student2//结构体变量名

定义了 student1, student2 为 struct student 类型的变量。

在定义了结构体变量后,系统会为之分配内存单元。例如 student1 和 student2在内存中各占59个字节。

应当注意,将一个变量定义为标准类型(基本数据类型)与定义为结构体类型不同之处在于后者不仅要求指定变量为结构体类型,而且要求指定为某一特定的结构体类型(例如 struct student 类型),因为可以定义出许多种具体的结构体类型。而在定义变量为整形时,只需指定为 int 型即可。

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

例如:

struct student

{

int num;

char name[20];

char sex;

int age;

float score;

char addr[30];

}student1, student2;

它的作用与第一种方法相同,即定义了两个 struct student 类型的变量 student1, student2 这种形式的定义的一般形式为

struct 结构体名

{

成员表列

}变量名表列;

(3)直接定义结构类型变量

其一般形式为

struct

{

成员表列

}变量名表列;

即不出现结构体名。

关于结构体类型,有几点要说明:

a. 类型与变量是不同的概念,不是混同,只能对变量赋值,存取或运算,而不能对一个类型赋值,存取或运算。在编译时,对类型是不分配空间的,只对变量分配空间。

b. 对结构体中的成员(即 域)可以单元使用,它的作用与地位相当于普通变量,

c. 成员也可以是一个结构体变量。

如:

struct date // 声明一个结构体类型

{

int month;

int day;

int year;

}

struct student

{

int num;

char name[20];

char sex;

int age;

struct date birthday;

char addr[30];

}student1, student2;

先声明一个 struct date 类型,它代表 日期 包括3个成员 month, day, year。然后在声明 struct student 类型时,将成员 birthday 指定为 struct date 类型。

d. 成员名可以与程序中的变量名相同,二者不代表同一对象。

3 结构体变量的引用

(1)不能将一个结构体变量作为一个整体进行输入和输出。

只能对结构体变量中的各个成员分别进行输入输出。引用结构体变量中的成员的方式为

结构体变量名.成员名

例如 student1.num 表示 student1 变量中的 num 成员,即 student1 的 num 项,可以对变量的成员赋值。例如:

student1.num = 10010;

"." 是成员(分量)运算符,它在所有的运算符中优先级最高,因此可以把 student1.num 作为一个整体来看待。上面的赋值语句作用是将整数 10010赋给 student1 变量中的成员 num。

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

例如:结构体变量 student1 可以这样访问各成员:

student1.num

student1.birthday.month

注意,不能用 student1.birthday 来访问 student1 变量中的成员 birthday,因为 birthday 本身是一个结构体变量。

(3)对结构体变量的成员可以像普通变量一样进行各种运算(根据其类型决定可以进行的运算)。

student2.score = student1.score;

sum = student1.score + student2.score;

student1.age ++;

++ student1.age;

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

(4)可以引用结构体变量成员的地址。也可以引用结构体变量的地址。如:

scanf("%d", &student1.num);// 输入 student1.num 的值

printf("%o", &student1);// 输出 student1 的首地址

但不能用以下语句整体读入结构体变量如:

scanf("%d,%s,%c,%d,%f,%s", &student1);

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

4 结构体变量的初始化

和其它类型变量一样,对结构体变量可以在定义时指定初始值。

如:

#include

struct student

{

long int num;

char name[20];

char sex;

char addr[30];

}a = {89031, "Li Lin", 'M', "123 Beijing Road"};

void main()

{

printf("NO. : %d\nname: %s\nsex: %c\naddress: %s\n", a.num, a.name, a.sex, a.addr);

}

5 结构体数组

一个结构体变量中可以存放一组数据(如一个学生的学号,姓名,成绩等数据)。如果有10个学生的数据需要参加运算,显然应该用数组,这就是结构体数组。结构体数组与以前介绍过的数据值型数组不同之处在于每个数组元素都一个结构体类型的数据,它们分别包括各个成员(分量)项。

5.1 定义结构体数组

和定义结构体变量的方法相仿,只需说明其为数组即可。

struct student

{

int num;

char name[20];

char sex;

int age;

float score;

char addr[30];

};

struct student stu[3];

以上定义了一个数组 stu,其元素为 struct student 类型数据,数组有 3 个元素。也可以直接定义一个结构体数组。如:

struct student

{

int num;

....

}stu[3];

struct

{

int num;

...

}stu[3];

5.2 结构体数组的初始化

与其它类型数组一样,对结构体数组可以初始化如:

struct student

{

int mum;

char name[20];

char sex;

int age;

float score;

char addr[30];

}stu[3] = {{10101,"Li Lin", 'M', 18, 87.5, "103 Beijing Road"},

{10101,"Li Lin", 'M', 18, 87.5, "103 Beijing Road"},

{10101,"Li Lin", 'M', 18, 87.5, "103 Beijing Road"}};

定义数组 stu 时,元素个数可以不指定,即写成以下形式:

stu[] = {{...},{...},{...}};

编译时,系统会根据给出初值的结构体常量的个数来确定数组元素的个数。

当然,数组的初始化也可以用以下形式:

struct student

{

int num;

...

};

struct student stu[] = {{...},{...},{...}};

即先声明结构体类型,然后定义数组为该结构体类型,在定义数组时初始化。

从以上可以看到,结构体数组初始化的一般形式是在定义数组的后面加上:

5.3 结构体数组应用举例

下面例子说明结构体数组的定义和引用。

#include

#include

#include

struct person

{

char name[20];

int count;

}leader[3] = {{"Li", 0},

{"Zhang", 0},

{"Fun", 0}};

void main()

{

int i, j;

char leader_name[20];

for(i = 1; i<= 10;i++)

{

scanf("%s", leader_name);

for(j=0;j<3;j++)

if(strcmp(leader_name, leader[j].name) == 0)

leader[j].count ++;

}

printf("\n");

for(i=0;i<3;i++)

printf("%5s: %d\n", leader[i].name, leader[i].count);

system("pause");

}

运行结果如下:

LI

Li

Fun

Zhang

Zhang

Fun

Li

Fun

Zhang

Li

Li: 3

Zhang: 3

Fun: 3

6 指向结构体类型数据的指针

一个结构体变量的指针就是该变量所占据的内存段的起始地址,可以设一个指针变量,用来指向一个结构体变量,此时该指针变量的值是结构体变量的起始地址。指针变量也可以用来指向结构体数组中的元素。

6.1 指向结构体变量的指针

指向结构体变量的指针的应用:

#include

#include

#include

struct student

{

long num;

char name[20];

char sex;

float score;

};

void main()

{

struct student stu_1;

struct student *p;

p = &stu_1;

stu_1.num = 89101;

strcpy(stu_1.name, "Li Lin");

stu_1.sex = 'M';

stu_1.score = 89.5;

printf("NO. :%ld\nname: %s\nsex:%c\nscore:%f\n", stu_1.num, stu_1.name, stu_1.sex, stu_1.score);

printf("NO. :%ld\nname: %s\nsex:%c\nscore:%f\n", (*p).num, (*p).name, (*p).sex, (*p).score);

system("pause");

}

在主函数中声明了 struct student 类型,然后定义了一个 struct student 类型的变量,stu_1 同时又定义一个指针变量 p ,它指向一个 struct student 类型的数据,在函数的执行部分将结构体变量 stu_1 的起始地址赋给指针变量 p ,也就是使 p 指向 stu_1 然后对 stu_1 的各成员赋值,第二个 printf 函数也是用来输出 stu_1 各成员的值,但使用的是 (*p).num 这样的形式, (*p) 表示 p 指向的结构体变量,(*p).num 是 p 指向的结构体变量中的成员 num 。注意 *p 两侧的括弧不可省略,因为成员运算符 '.' 优先于 '*' 运算符,*p.num 就等价于 *(p.num)

运行结果如下:

NO. :89101

name: Li Lin

sex:M

score:89.500000

NO. :89101

name: Li Lin

sex:M

score:89.500000

可以看到两个 printf 输出的结果相同。

在C语言中,为了使用方便和使之直观,可以把 (*p).num 改用 p->num 来代替,它表示 *p 所指向的结构体变量中的 num 成员,同样,(*p).name 等价于 p->name。

也就是说以下三种形式等价:

a. 结构体变量.成员名

b. (*p).成员名

c. p->成员名

上面的最后一个 printf 函数输了项可以改写为

printf("NO. :%ld\nname: %s\nsex:%c\nscore:%f\n",p->num, p->name, p->sex, p->score);

其中 -> 称为指向运算符。

分析以下几种运算符

p -> n 得到 p 指向的结构体变量中的成员 n 的值

p -> n ++ 得到 p 指向的结构体变量中的成员 n 的值,用完值后使它加1

++p -> n 得到 p 指向的结构体变量中的成员 n 的值使之加 1 (先加)

6.2 指向结构体数组的指针

以前介绍过可以使用指向数组或数组元素的指针和指针变量,同样,对结构体数组及其元素也可以用指针变量来指向。

指向结构体数组的指针的应用

#include

#inlcude

struct student

{

int num;

char name[20];

char sex;

int age;

};

struct student stu[3] = {{10101, "Li Lin", 'M', 18},

{10102, "Zhang Fun", 'M', 19},

{10103, "Wang Min", 'F', 20}};

void main()

{

struct student *p;

printf("No.  name    sex    age\n");

for(p=stu; p

printf("%5d %-20s %2c %4d\n", p->num, p->name, p->sex, p->age);

system("pause");

}

运行结果如下:

No.  name    sex    age

10101 Li Lin                M     18

10102 Zhang Fun        M     19

10103 Wang Min          F      20

注意以下两点:

(1)如果 p 的初值为 stu,即指向第一个元素,则 p + 1 后指向下一个元素的起始地址。例如:

(++p) -> num 先使 p 自加 1 ,然后得到它指向的元素中的 num 成员的值(即10102)。

(p++) ->num 先得到 p->num 的值(即10101),然后使 p 自加 1 ,指向 stu[1]。

注意以上二者的不同。

(2)程序已定义了指针 p 为指向 struct student 类型数据的变量,它只能指向一个 struct student 型的数据(p 的值是 stu 数组的一个元素的起始地址),而不能指向 stu 数组元素中的某一成员,(即 p 的地址不能是成员地址)。例如,下面是不对的:

p = &stu[1].name

编译时将出错。千万不要认为反正 p 是存放地址的,可以将任何地址赋给它。如果地址类型不相同,可以用强制类型转换。例如:

p = (struct student *)&stu[1].name;

此时,在 p 中存放 stu[1] 元素的 name 成员的起始地址。

6.3 用结构体变量和指向结构体的指针作函数参数

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

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

(2)用结构体变量作参数。老版本的C系统不允许用结构体变量作实参,ANSI C取消了这一限制。但是用结构体变量作实参时,采取的是 值传递 的方式,将结构体变量所占的内存单元全部顺序传递给形参。形参也必须是同类型的结构体变量。在函数调用期间形参也要占用内存单元。这种传递方式在空间和时间上开销较大,如果结构体的规模很大时,开销是很可观的,此外由于采用值传递方式,如果在执行被调用函数期间改变了形参(也是结构体变量)的值,该值不能返回主调函数,这往往造成使用上的不便。因此一般较少用这种方法。

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

用结构体变量作函数参数。

#include

#define FORMAT "%d\n%s\n%f\n%f\n%f\n"

struct student

{

int num;

char name[20];

float score[3];

};

void print(struct student stu)

{

printf(FORMAT, stu.num, stu.score[0], stu.score[1], stu.score[2]);

printf("\n");

}

void main()

{

struct student stu;

stu.num = 12345;

strcpy(stu.name, "Li Li");

stu.score[0] = 67.5;

stu.score[1] = 89;

stu.score[2] = 78.6;

printf(stu);

}

将上面改用指向结构体变量的指针作实参。

#include

#define FORMAT "%d\n%s\n%f\n%f\n%f\n"

struct student

{

int num;

char name[20];

float score[3];

}stu = {12345, "Li Li", 67.5, 89, 78.6};

void print(struct student *p)

{

printf(FORMAT, p->num, p->name, p->score[0], p->score[1], p->score[2]);

printf("\n");

}

void main()

{

print(&stu);

}

赞赏

共11人赞赏

本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。

c语言中 结构体对象,C语言中结构体详解相关推荐

  1. java中的model_Java程序员必看之springmvc中的Model对象在重定向中的数据

    原标题:Java程序员必看之springmvc中的Model对象在重定向中的数据 在springmvc框架中,一个handler方法中的参数可以内置接收Model类型的对象,主要用于存储数据之用,主要 ...

  2. 编程题50 习题6-8 单词首字母大写【浙大版《C语言程序设计(第4版)》题目集 详解教程】

    编程题50 习题6-8 单词首字母大写[浙大版<C语言程序设计(第4版)>题目集 详解教程] 原题链接:习题6-8 单词首字母大写 (pintia.cn) 参考答案 #include< ...

  3. [ASP.NET]ASP.NET中的Inherits、CodeFile、CodeBehind的区别详解及OWIN及IDENTITY

    篇一 : ASP.NET中的Inherits.CodeFile.CodeBehind的区别详解 Inherits.CodeFile.CodeBehind 在 ASP.NET 中使用代码隐藏方法来设计W ...

  4. linux mv复制命令,linux中删除复制移动文件rm,mv,cp命令详解linux操作系统 -电脑资料...

    在linux中对文件的复制删除移动分别会使用到rm,mv,cp三个命令,下面我来给大家介绍一下rm,mv,cp命令对文件的常规操作吧, 先看实例 删除复制移动文件命令 Linux代码 rm -rf / ...

  5. php中单引号,php中的单引号、双引号和转义字符详解

    PHP单引号及双引号均可以修饰字符串类型的数据,如果修饰的字符串中含有变量(例$name):最大的区别是: 双引号会替换变量的值,而单引号会把它当做字符串输出. 例如: 结果: 字符串$name 字符 ...

  6. python反向缩进_在Pycharm中对代码进行注释和缩进的方法详解

    在Pycharm中对代码进行注释和缩进的方法详解 一.注释 1. #单行注释 2. """ 多行注释 """ 3. pycharm多行注释快 ...

  7. python中怎么计数_浅谈python中统计计数的几种方法和Counter详解

    1) 使用字典dict() 循环遍历出一个可迭代对象中的元素,如果字典没有该元素,那么就让该元素作为字典的键,并将该键赋值为1,如果存在就将该元素对应的值加1. lists = ['a','a','b ...

  8. python实现单例模式的几种方式_基于Python中单例模式的几种实现方式及优化详解...

    单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在.当你希望在整个系统中,某个类只能出现一个实例时,单例对象就能派上用场. ...

  9. python中with open写csv文件_Python中的CSV文件使用with语句的方式详解

    是否可以直接使用with语句与CSV文件?能够做这样的事情似乎很自然: import csv with csv.reader(open("myfile.csv")) as read ...

  10. 处理对象(toString()方法详解和==与equals方法的区别)

    处理对象(toString()方法详解和==与equals方法的区别) toString()是一个非常特殊的方法,它是一个自我描述的方法.当程序员直接打印该对象的时候,系统会输出该对象的"自 ...

最新文章

  1. C# 写Windows服务
  2. Java NIO 学习笔记 缓冲区补充
  3. C#学习小记12实现一个接口
  4. jvm性能调优实战 - 40 百万级数据误处理导致的频繁Full GC问题优化
  5. 将.NET dll注册到GAC(Global Assembly Cache)中
  6. Pytorch高阶API示范——线性回归模型
  7. java as操作_java 对类型的基本操作小结
  8. 黑客是互联网的高危职业,选择不慎,很可能如“他们”一样入门到入狱!
  9. n条直线相交最多有几个邻补角_【初中数学】102条做初中几何辅导线的规律
  10. 欲练JS,必先攻CSS——前端修行之路
  11. 布同:vim插件开发实例和讲解
  12. java sha1工具类_SHA1算法工具类
  13. 【Java基本功】很多人经常忽视的Java基础知识点
  14. tomcat设置为开机自启动
  15. 24位掩码和30个掩码_高级ds位掩码和dp的问题
  16. 【教程6】疯壳·ARM功能手机-中断实验教程
  17. 美丽的日本与我(川端康成在诺贝尔文学奖颁奖典礼上的演讲词)
  18. 第一篇:详细介绍三次握手和四次挥手
  19. 关于设计的一点思考:
  20. 只会Python可造不出iPhone

热门文章

  1. xLang 的类型转换
  2. matlab 线透明度,matlab 曲线填充 透明度
  3. discuz文件结构详解,discuz模板文件介绍
  4. Windows中MSOCache文件夹
  5. Office Professional Plus 2010 产品密钥
  6. 面试官;经典面试题-JVM篇
  7. 奥克兰理工大学计算机学院,9月17日学术报告(新西兰奥克兰理工大学 Prof. Re
  8. 奥克兰大学计算机专业世界排名,奥克兰大学,15学科排名世界前50!
  9. java lazy实现依赖调用_spring 之 lazy-init Autowired depends-on
  10. React构建的景点门票系统