梓益C语言学习笔记之结构体-共用体-枚举
有人说,我们需要将不同类型的数据组合成一个整体,于是,就有了结构体
一、结构体的概念:
一种或多种基本类型或构造类型的数据的集合。
二、定义方法:
先定义结构体类型,再去定义结构体变量
struct 结构体类型名{
成员列表
};
如:
struct stu{
int num;
char name[20];
char sex;
};
struct stu lucy,bob,lilei;//定义了三个struct stu类型的变量
每个变量都有三个成员,分别是num name sex
通常咱们将一个结构体类型重新起个类型名,用新的类型名替代原先的类型
**最常用的定义结构体的方法是:
typedef struct stu{
int num;
char name[20];
char sex;
}STU;
STU lucy;
三、初始化及使用:
typedef struct stu{
int num;
char name[20];
char sex;
}STU;
STU lucy={101,"lucy",'f'};
STU bob;
bob.num=101;//bob是个结构体变量,但是bob.num是个int 类型的变量
bob.name 是个字符数组,是个字符数组的名字,代表字符数组的地址,是个常量
bob.name ="bob";//是不可行,是个常量
strcpy(bob.name,"bob");
四、结构体数组
结构体数组是个数组,由若干个相同类型的结构体变量构成的集合
1、结构体数组的定义方法:
struct stu{
int num;
char name[20];
char sex;
};
struct stu edu[3];//定义了一个struct stu 类型的结构体数组edu,
这个数组有3个元素分别是edu[0] 、edu[1]、edu[2]
可以这样赋值:STU edu[3]={
{101,"Lucy",78},
{102,"Bob",59.5},
{103,"Tom",85}
};
2、数组元素的使用
edu[0].num =101;//用101给edu数组的第0个结构体变量的num赋值
strcpy(edu[1].name,"lucy");
五、结构体指针
即结构体的地址,结构体变量存放内存中,也有起始地址
定义一个变量来存放这个地址,那这个变量就是结构体指针变量。
1、结构体指针变量的定义方法:
struct stu{
int num;
char name[20];
};
struct stu * p;//定义了一个struct stu *类型的指针变量,变量名是p,p占4个字节,用来保存结构体变量的地址编号
使用: struct stu boy;
p=&boy;
访问结构体变量的成员方法:
boy.num=101;//可以,通过结构体变量名.成员名
(*p).num=101;//可以,*p相当于p指向的变量boy
p->num=101;//可以,指针->成员名
通过结构体指针来引用指针指向的结构体的成员,前提是指针必须先指向一个结构体变量。
六、结构体内存分配
为什么要有字节对齐?
用空间来换时间,提高cpu读取数据的效率
结构体变量大小是所有成员之和
规则1:结构体分配内存时哪个基本类型的成员占字节数多,就以它的大小为单位开辟内存,
在gcc中出现了double类型的,例外
1:成员中只有char型数据 ,以1字节为单位开辟内存。
2:成员中出现了short int 没有更大字节数的基本类型数据以2字节为单位开辟内存
3、出现了int float 没有更大字节的基本类型数据的时候以4字节为单位开辟内存。
4、出现了double类型的数据
情况1:
在vc里,以8字节为单位开辟内存。
情况2:
在gcc里,以4字节为单位开辟内存。
无论是那种环境,double型变量,占8字节。
注意:
如果在结构体中出现了数组,数组可以看成多个变量的集合。
如果出现指针的话,没有占字节数更大的类型的,以4字节为单位开辟内存。
在内存中存储结构体成员的时候,按定义的结构体成员的顺序存储。
例1:struct stu{
char sex;
int age;
}lucy;
lucy的大小是4的倍数。
规则2:字节对齐
1:char 1字节对齐 ,即存放char型的变量,内存单元的编号是1的倍数即可。
2:short int 2字节对齐 ,即存放short int 型的变量,起始内存单元的编号是2的倍数即可。
3:int 4字节对齐 ,即存放int 型的变量,起始内存单元的编号是4的倍数即可
4:long int 在32位平台下,4字节对齐 ,即存放long int 型的变量,起始内存单元的编号是4的倍数即可
5:float 4字节对齐 ,即存放float 型的变量,起始内存单元的编号是4的倍数即可
6:double
1.vc环境下
8字节对齐,即存放double型变量的起始地址,必须是8的倍数,double变量占8字节
2.gcc环境下
4字节对齐,即存放double型变量的起始地址,必须是4的倍数,double变量占8字节。
注意3:当结构体成员中出现数组的时候,可以看成多个变量。
注意4:开辟内存的时候,从上向下依次按成员在结构体中的位置顺序开辟空间
例1:
struct stu{
char a;
short b;
int c;
}temp;
例3://temp 8个字节
#include
struct stu{
char a;
short int b;
int c;
}temp;
结果分析:
a的地址和b的地址差2个字节
b的地址和c的地址差2个字节
例4:temp的大小为12个字节
#include
struct stu{
char a;
int c;
short int b;
}temp;
结果分析:
a和c的地址差4个字节
c和b的地址差4个字节
例5:
struct stu{
char buf[10];
int a;
}temp;
//temp占 16个字节
例6:
struct stu{
char a;
double b;
}temp;
temp:
在vc中占 16个字节 a和b的地址差8个字节
在gcc中占 12个字节 a和b的地址差4个字节
#include
struct stu{
char a;
double b;
}temp;
指定对齐原则:
使用#pragma pack改变默认对其原则
#pragma pack (value)
1.value只能是:1 2 4 8等
2.指定对齐值与数据类型对齐值相比取较小值
说明:咱们制定一个value
1:以多少个字节为单位开辟内存
结构体成员中,占字节数最大的类型长度和value比较,
取较小值,为单位开辟内存
例7:
#pragma pack(2)
struct stu{
char a;
int b;
} ;
以2个字节为单位开辟内存
#include
#pragma pack(2)
struct stu{
char a;
int b;
}temp;
int main()
{
printf("%d\n",sizeof(temp));
printf("%p\n",&(temp.a));
printf("%p\n",&(temp.b));
return 0;
}
temp的大小为6个字节
a和b的地址差2个字节
如:如果指定对齐值:
设为1:则short、int、float等均为1
设为2:则char仍为1,short为2,int 变为2
七、位段
在结构体中,以位为单位的成员,咱们称之为位段(位域)。
struct packed_data{
unsigned int a:2;
unsigned int b:6;
unsigned int c:4;
unsigned int d:4;
unsigned int i;
} data;
对于位段成员的引用如下:
data.a =2
赋值时,不要超出位段定义的范围;
如段成员a定义为2位,最大值为3,即(11)2
所以data.a =5,就会取5的低两位进行赋值 101
八、共用体
**每次只能使用共用体的一个变量
几种不同类型的变量存到同一段内存单元中,几个变量所使用空间相互重叠
这种几个不同的变量共同占用一段内存的结构,在C语言中,被称作“共用体”类型结构
共用体所有成员占有同一段地址空间
共用体的大小是其占内存长度最大的成员的大小
typedef union data{
short int i;
char ch;
float f;
}DATA;
DATA temp2;
共用体temp2占4个字节,即i、ch、f共用4个字节
九、枚举
将变量的值一一列举出来,变量的值只限于列举出来的值的范围内
例:
定义枚举类型 week
enum week //枚举类型
{
mon,tue,wed,thu,fri,sat,sun
};
enum week workday,weekday;//枚举变量
workday与weekday只能取sun….sat中的一个