c语言 结构体的,c语言之结构体
c语言之结构体
1.结构体基础知识
C语言提供了两种类型的聚合数据类型(能够同时存储超过一个的单独数据),数组和结构。数则是相同类型的元素的集合,而结构也是一些值的集合,这些值称为它的成员,单一个结构的各个成员可能具有不同的类型。
由于结构成员不一定是同种数据类型,所以不能用类似于数组的引用方式,而是用成员名来引用。
1.1结构声明:
struct tag {member-list} variable-list;//注意这里的分号
sruct item
{
int a;
char b;
float c;
}v1,v2={1,'2',3.0};
这里声明了两个变量v1,v2,v2的成员并初始化了。
1.2结构引用:
sruct item *p;
(1)指针变量:p->a;p-b;(*p).va
(2)普通变量:v1.a;v2.b;(&v1)->a
(3)结构的自引用:
非法:sruct item p;注意这里声明的是一个结构变量,如果这样声明会类似于递归无休止的下去,非法的。
sruct item
{
int a;
char b;
float c;
sruct item p;
};
合法:sruct item *p;注意这里分配的是一个指针,指针是固定大小的,需要用的时候分配空间。
sruct item
{
int a;
char b;
float c;
sruct item *p;
};
1.3结构的存储分配
系统并不像结构里面所定义的变量分配相应的空间,结构所占的空间大小还要算上由于边界对齐(数据对齐)所带来的额外内存空间,为什么会出现数据对齐呢?
8位的CPU当然不会产生数据对齐,但是发展到16位,32位时就会产生,以32位的为例,CPU一次能够内存访问的是4个字节的数据量,那么如果一个1字节的数据存储在里面当然就没有问题,但是如果2字节,或者4字节的数据存储时候超过了边界,跨越了两个4的倍数的单位,那么就要读取两次了。
例如,下面的结构各成员空间分配情况:struct test
{
char x1;
short x2;
float x3;
char x4;
};
结构的第一个成员x1,其偏移地址为0,占据了第1个字节。第二个成员x2为short类型,其起始地址必须2字节对界,因此,编译器在x2和x1之间填充了一个空字节。结构的第三个成员x3和第四个成员x4恰好落在其自然对界地址上,在它们前面不需要额外的填充字节。在test结构中,成员x3要求4字节对界,是该结构所有成员中要求的最大对界单元,因而test结构的自然对界条件为4字节,编译器在成员x4后面填充了3个空字节。整个结构所占据空间为12字节。
如果我们改变其结构内部的顺序则可以改变其整体大小,
struct test
{
char x1;
char x4;
short x2;
float x3;
};
Sizeof操作符能够得出一个结构的整体大小,包括因边界对齐而跳过的部分,宏offsetof(定义于stddef.h)可以得到每个成员的具体位置,
Offsetof(type, member)
例如:offsetof(struct test, x1),struct test为结构体,x1为结构里面的成员名
测试程序:
#include
#include
struct test1
{
char x1;
short x2;
float x3;
char x4;
};
struct test2
{
char x1;
char x4;
short x2;
float x3;
};
int main(void)
{
printf("test 1 size is %d\n", sizeof(struct test1));
printf("x1 offset is %d\n", offsetof(struct test1, x1));
printf("x2 offset is %d\n", offsetof(struct test1, x2));
printf("x3 offset is %d\n", offsetof(struct test1, x3));
printf("x4 offset is %d\n", offsetof(struct test1, x4));
printf("test 2 size is %d\n", sizeof(struct test2));
printf("x1 offset is %d\n", offsetof(struct test2, x1));
printf("x4 offset is %d\n", offsetof(struct test2, x4));
printf("x2 offset is %d\n", offsetof(struct test2, x2));
printf("x3 offset is %d\n", offsetof(struct test2, x3));
return 0;
}
程序结果:
test 1 size is 12
x1 offset is 0
x2 offset is 2
x3 offset is 4
x4 offset is 8
test 2 size is 8
x1 offset is 0
x4 offset is 1
x2 offset is 2
x3 offset is 4
请按任意键继续. . .
边界对齐的规则:
第一,编译器按照成员列表的顺序给每个成员分配内存.第二,当成员需要满足正确的边界对齐时,成员之间用额外字节填充.第三,结构体的首地址必须满足结构体中边界要求最为严格的数据类型所要求的地址.第四,结构体的大小为其最宽基本类型的整数倍.
#pragma pack(n)指令
在缺省情况下,C编译器为每一个变量或是数据单元按其自然对界条件分配空间。一般地,可以通过下面的方法来改变缺省的对界条件:·使用伪指令#pragma pack (n),C编译器将按照n个字节对齐。·使用伪指令#pragma pack (),取消自定义字节对齐方式。
1.4结构作为函数参数:由于结构作为函数参数,一般结构大小较大,直接作为函数参数效率较低,会大大降低栈的大小(函数参数放于栈上),我们一把采用的是指针传递,这样就需要注意指针使用的安全性,必要的时间尽量加上限定符const
例如:
void print(struct test1 mytest)
{
printf("x1 = %c, x3 = %f\n",mytest.x1, mytest.x3);
}
void print(struct test1 *mytest)
{
printf("x1 = %c, x3 = %f\n",mytest->x1, mytest->x3);
}
由于函数内部不改变其值,所有我们应该限定为指针指向的为常量
void print(struct test1 *constmytest)
{
printf("x1 = %c, x3 = %f\n",mytest->x1, mytest->x3);
}
2.1typedef与结构的联合应用
typedef与define的区别:
我们经常看到这样的用法:
typedef struct node
{
char x1;
char x4;
short x2;
float x3;
}NODE, *PNODE;
我们可以拆开了慢慢讲,上面的定义顺序完成的是
(1)首先是结构体得定义
struct node
{
char x1;
char x4;
short x2;
float x3;
};
结构名为node,我们可以利用结构名定义结构变量;然后就是下一步进行的是
(2)typedef struct node NODE;//将struct node创建一个名字为NODE
typedef struct node * PNODE;//将struct node*创建一个名字为PNODE
需要注意的是,在C99标准里面,不将struct node写完全是不能通过编译的(现在可以写成node,注意在C语言里面这个好像也不能通过编译的,估计就是该编译器使用的是C99标准,但是在C++里面可以通过编译),我们应该养成好习惯.。
如果我们在声明结构体的时候忽略结构名得话,必须在声明的时候定义变量不然后面将无法使用该结构了,但是你也可以使用,比如:
typedef struct
{
char x1;
char x4;
short x2;
float x3;
}mNODE, *pnode;
但是如果你没有上面定义别名,那么你只能按下面的方法声明并定义变量
struct
{
char x1;
char x4;
short x2;
float x3;
}node1;
(3)由于定义顺序是按照(1)(2),进行的,所有下面的代码是通不过编译的
typedef struct node
{
char x1;
char x4;
short x2;
float x3;
PNODE l;
}NODE, *PNODE;
(4)需要注意的是结构体指针,我们在定义结构指针的时候一定要初始化为NULL,使用的使用必须给他动态分配空间,除非已经有结构体变量了,你可以指针指向它,这就不需要分配动态空间。
(5)假如已经有了结构体struct node,我们定义
typedef struct node NODE
typedef NODE * PNODE等同于typedef struct node * PNODE
其实我们为了更具有阅读性,我们应该选择更易理解的定义方式,而不是写出很具有艺术性的代码,让大家都不是很容易懂。
(6)自定义结构数组:如果我们自定义结构数组很容易像下面这样使用:
typedef struct node[10] ANODE;
正确的用法是:
typedef struct nodeANODE [10];
或者
typedef struct node
{
char x1;
char x4;
short x2;
float x3;
}NODE, *PNODE, ANODE [10];
定义变量结构数组:
ANODE array1;//代表的是struct node array1[10]
ANODE array2[2];//代表的是struct node array2[2][10]
测试用例:
#include
#include
typedef struct node
{
char x1;
char x4;
short x2;
float x3;
}NODE, *PNODE, ANODE [10];
ANODE array1;
ANODE array2[2];
int main(void)
{
int a[2][10];
printf("struct node size is %d\n", sizeof(struct node));
printf("array1 size is %d\n", sizeof(array1));
printf("array2 size is %d\n", sizeof(array2));
printf("array2[0] size is %d\n", sizeof(array2[0]));
printf("array2[0][3] size is %d\n", sizeof(array2[0][3]));
printf("array2[2][3] size is %d\n", sizeof(array2[2][3]));
printf("a[0][0] offset is %d\n", &a[0][0]);
printf("a[0][9] offset is %d\n", &a[0][9]);
printf("a[1][0] offset is %d\n", &a[1][0]);
printf("a[1][9] offset is %d\n", &a[1][9]);
printf("a[0] offset is %d\n", &a[0]);
printf("a[1] offset is %d\n", &a[1]);
printf("array1[1] offset is %d\n", &array1[1]);
printf("array1[0] offset is %d\n", &array1[0]);
printf("array2[1] offset is %d\n", &array2[1]);
printf("array2[0] offset is %d\n", &array2[0]);
return 0;
}
程序结果:
struct node size is 8
array1 size is 80
array2 size is 160
array2[0] size is 80
array2[0][3] size is 8
array2[2][3] size is 8
a[0][0] offset is 37814032
a[0][9] offset is 37814068
a[1][0] offset is 37814072
a[1][9] offset is 37814108
a[0] offset is 37814032
a[1] offset is 37814072
array1[1] offset is 4206760
array1[0] offset is 4206752
array2[1] offset is 4206672
array2[0] offset is 4206592
请按任意键继续. . .
结果分析:我们可以看出&a[1]-&a[0]是10个整型的大小,因为我们定义的是int a[2][10];
现在我可以用类似的方法测试ANODE array2[2];到底是array2[2][10],还是array2[10][2];
由结果
array2[1] offset is 4206672
array2[0] offset is 4206592
我们可以知道,ANODE array2[2]定义的是array2[2][10],这是二维的数组,多维数组可以用类似的方法测试。
c语言 结构体的,c语言之结构体相关推荐
- go var type 互转_Go语言学习笔记(第九章) 结构体
Go语言基础之结构体 Go语言中没有"类"的概念,也不支持"类"的继承等面向对象的概念.Go 通过类型别名(alias types)和结构体的形式支持用户自定义 ...
- c语言结构体与共同体课件,《结构体与共同体》PPT课件.ppt
<<结构体与共同体>PPT课件.ppt>由会员分享,可在线阅读,更多相关<<结构体与共同体>PPT课件.ppt(44页珍藏版)>请在装配图网上搜索. 1 ...
- c语言中有关void,sizeof,结构体的一些问题
void[1]: void是C语言中的空类型,void的用途有二. 1.对函数返回的限定: 如果函数没有返回值,则默认返回整数类型,而不是void类型.c++有很严格的类型,不允许函数不加类型声明,而 ...
- 【精华文】C语言结构体特殊情况分析:结构体指针 / 基本数据类型指针,指向其他结构体
参考链接:Structure pointer pointing to different structure instance 注:可以查看此篇的问题和唯一的回复,那是相对正确的,不要看comment ...
- 结构体怎么赋值_c语言学习之基础知识点介绍:结构体的介绍
一.结构体的介绍 /* 语法:struct 结构体名{成员列表;};切记切记有分号!说明:成员列表就是指你要保存哪些类型的数据.注意:上面的语法只是定义一个新的类型,而这个类型叫做结构体类型.因为类型 ...
- python中的记录指针_使用Python向C语言的链接库传递数组、结构体、指针类型的数据...
使用python向C语言的链接库传递数组.结构体.指针类型的数据 由于最近的项目频繁使用python调用同事的C语言代码,在调用过程中踩了很多坑,一点一点写出来供大家参考,我们仍然是使用ctypes来 ...
- c语言定义学生结构体类型,C语言中结构体的三种定义方式
c语言中结构体的定义: struct 结构体名{ 成员列表: ..... }结构体变量: 7.1.1 结构体类型变量的定义 结构体类型变量的定义与其它类型的变量的定义是一样的,但由于结构体类型需要针对 ...
- c语言有2维结构体没,c语言结构体说明
关键词:语言,结构 摘要:1.直接声明结构体变量: struct{ int length; int width; }box1; 这样就声明了一个名为box1的结构体变量,但是同时要注意,如果在同一个程 ...
- c语言 结构体声明和引用、,结构体的声明与自引用
今天上了数据结构课程的第一堂课,经常会看到下面这种语法:结构体有个成员变量是指向该结构体的指针,也就是自引用(self reference).翻看了下一章节内容,才知道这是链表的结构基础.平时C语言用 ...
最新文章
- PHP操作MYSQL--PDO
- 工作109:路由菜单项
- oracle其他盘添加表空间吗,oracle增加表空间的四种方法
- c语言中通过指针引用数组,C语言基础(二)
- 为什么要学习Python?用数据给你展示八大理由!!
- 奥特曼传奇英雄存档丢了怎么找回_热血传奇:法师前期跨级刷怪到底划不划算?...
- Java学习之路 之 使用技巧篇
- 【指纹识别】基于matlab GUI指纹打卡系统【含Matlab源码 867期】
- c++ 取整_MPIP Raw转Raw图简述-C实现
- 【SSH密钥生成与使用】
- Oracle彻底卸载干净教程
- python控制小爱同学_GitHub - xinruoyusixian/Blinker-for-xiaoai: 基于micopython 的小爱同学的代码...
- 解决win7英文版下中文显示乱码问题
- linux下lamealsa进行音频流操作(四)alsa+lame将音频流转为MP3
- MySQL数据库高级SQL语句【进阶查询、null值、数据库正则、数据库运算符、连接查询(内连接、左外连接、右外连接)】
- Android Q适配攻略(五)(存储权限变更)
- python练习题:程序员问卷调查
- centos/redhat kernel-debug-info-xx.rpm与kernel-debuginfo-xx.rpm区别
- VUE进阶篇Part9(render函数)
- 使用html和js制作饼图,使用js画图之饼图
热门文章
- Quartz CronTrigger时间最完整配置说明
- 文件传输服务器的默认端口,服务器上传文件端口
- fastreport文本字数太多换行_Python教程第10篇:聊聊print换行输出和重复多次打印...
- 设计灵感|展览海报如何编排?
- UI设计师必备知识|最全UI设计规范!
- 手机移动端APP引导界面设计模板素材,分层可临摹
- mysql 安装 权限_MySQL的安装、使用及权限管理
- 《Web前端开发最佳实践》学习笔记
- Windows获取系统版本号
- Linux网络防火墙:iptables与netfilter