Bootstrap

C语言——结构体

1 结构体初识篇

C语言的结构体其实就相当于C++里的类

#include <stdio.h>
#include <string.h>

1.创建结构体
结构体类型-->描述结构化数据
	struct 标识符
 {
	  特征--->基本数据类型描述
 };

struct  MM 
{
	char name[20];
	int age;
	int num;
	char addr[20];
};
1.1 C语言不允许空的结构体存在
	//struct empty 
	//{
	//};

-----------------------------------------------------------------------------------------
    
2.怎么访问数据
2.1 结构体中数据必须要通过结构体变量访问
2.2 怎么创建结构体变量  类型+变量名;
2.3 访问方式
	 结构体变量  变量.成员
	 结构体指针  指针->成员
int main() 
{	
	//创建结构体变量
	struct MM  girl;
    
	//数组不能不能直接赋值需要strcpy
	//girl.name = "sdfg";
	strcpy_s(girl.name, 20, "小美");
	girl.age = 38;
	//指针用->  一般不这样写
	(&girl)->num = 1001;
	strcpy_s(girl.addr, 20, "我家");
	printf("%s\t%d\t%d\t%s\n", girl.name, girl.age, girl.num, girl.addr);
    
    
	//类型* 变量名;
	struct MM* pMM = NULL;
	//有问题 NULL里面没有age
	//pMM->age = 123;(错误)
	pMM = &girl;
	printf("%s\t%d\t%d\t%s\n", (*pMM).name, (*pMM).age, (*pMM).num, (*pMM).addr);
	//->  指针指向运算符   这种好看
	printf("%s\t%d\t%d\t%s\n", pMM->name, pMM->age, pMM->num, pMM->addr);

	return 0;
}

2 结构体的创建与初始化

#include <stdio.h>
struct MM 
{
	char name[20];
	int age;
	double score;
};
-----------------------------------------------------------------------------------------
2.结果体变量创建
struct MM g_mm = { "全局",18,1.2f };//先创建再初始化
struct Data 
{
	char name[20];
	int age;
	double score;
}value = {"全局",18,1.0f},arr[3],a,b;
//value是变量名
//value ,a,b 都是一个结构体变量
//arr[3] 结构体数组

-----------------------------------------------------------------------------------------

3.typedef 与结构体创建(可以简化代码,目前觉得鸡肋,代码长了就有用了)
typedef struct Test 
{
	char name[20];
	int age;
	int score;
}Test,*PTest,ARRAY[3];
//Test,*PTest,ARRAY[3] 都是类型别名
//typedef struct Test Test,
//typedef struct Test *PTest,
//typedef struct Test ARRAY[3];
void  test_typedef_struct() 
{
	struct Test t1;
	Test t2;			//struct Test t2
	struct Test* p1 = NULL;
	PTest p2 = NULL;	// struct Test *p2

	struct Test array1[3];
	ARRAY array2;		// struct Test array2[3]
}
-----------------------------------------------------------------------------------------
    
4 特殊结构体定义
struct  //没有名字的结构体这个结构体只有一个变量
{
	int num;
	int key;
}SingleTon;

typedef struct //有typedef,就有了别名,可以随意创建变量
{
	int num;
	int key;
}Type;
void test() 
{
	Type a = { 1,2 };
	Type b = { 2,3 };//可以无限定义变量
	SingleTon.num = 23;
	SingleTon.key = 123;//只有这两个
}

-----------------------------------------------------------------------------------------

int main() 
{
	//1.可以创建变量的时候用大括号方式初始化(局部的)
	struct MM mm1 = { "小帅",18,1.234 };
	struct MM mm2 = { "小美"};	//age和score默认为0
	struct MM mm3 = { .age = 12,.name = "小花",.score = 1.2f };
	printf("%d\t%.2lf\n", mm2.age, mm2.score);
	printf("%s\t%d\t%.1lf\n", mm3.name, mm3.age, mm3.score);
	
    test_typedef_struct() 
        
    //测试3    
    struct black_red_node* p = (struct black_red_node*)malloc(sizeof(struct black_red_node));
	RBNode* p1 = (RBNode*)malloc(sizeof(RBNode));    
	return 0;
}

3 结构体数组

#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <string.h>
struct MM 
{
	char name[20];
	int age;
	char addr[20];
};
//1 输入和打印
void test() 
{
	struct MM arr[3];
	printf("input mm(3):");
	for (int i = 0; i < 3; i++)
	{
		//scanf("%s%d%s", arr[i].name,&arr[i].age, arr[i].addr);
		scanf_s("%s%d%s", arr[i].name, 20, &arr[i].age, arr[i].addr, 20);//scanf_s要加尺																		寸,别的编译器不用
	}
	printf("姓名\t年龄\t编号\n");
	for (int i = 0; i < 3; i++)
	{
		printf("%s\t%d\t%s\n", arr[i].name, arr[i].age, arr[i].addr);
	}
}
-----------------------------------------------------------------------------------------
    
//2 结构体数组的冒泡排序(传函数指针的方式)    
void bubble_sort(struct MM* arr, int arrayNum,bool(*compare)(struct MM,struct MM)) 
{
	for (int i = 0; i < arrayNum; i++) 
	{
		for (int j = 0; j < arrayNum - i - 1; j++) 
		{
			//比较规则,剥洋葱
			if (compare(arr[j],arr[j+1]))
			{
				//memcpy
				struct MM temp = arr[j];
				arr[j] = arr[j + 1];
				arr[j + 1] = temp;
			}
		}
	}
}
void print_mm(struct MM arr[], int arrayNum) 
{
	printf("姓名\t年龄\t编号\n");
	for (int i = 0; i < arrayNum; i++)
	{
		printf("%s\t%d\t%s\n", arr[i].name, arr[i].age, arr[i].addr);
	}
}
//可以自己写按什么方式去比较(用函数指针升级一下)
bool compare_by_age(struct MM one, struct MM two) 
{
	return one.age > two.age;
}

bool compare_by_name(struct MM one, struct MM two)
{
	return strcmp(one.name,two.name)>0;
}
bool compare_by_addr(struct MM one, struct MM two)
{
	return strcmp(one.addr, two.addr)>0;
}

-----------------------------------------------------------------------------------------

int main() 
{
	struct MM data[5] = 
	{  "4小芳",18,"6地球",
		"2小花",48,"776号" ,
		"1小美",28,"8海南" ,
		"4小李",38,"9天津" ,
		"6小小",58,"0西藏"
	};
	bubble_sort(data, 5,compare_by_age);
	print_mm(data, 5);
	bubble_sort(data, 5, compare_by_name);
	print_mm(data, 5);
	bubble_sort(data, 5, compare_by_addr);
	print_mm(data, 5);
	return 0;
}

4 位段

以后可能会看见,但自己很少用(做特别精小的软硬件)

#include <stdio.h>
#include <stdlib.h>
struct Flag 
{
	//:后面的数字代表的是变量占用内存是多少个二进制位
	unsigned int a : 3;	//3个二进制位   _ _ _  最小000  最大111
	unsigned int b : 2;	//2个二进制位   _ _    最小00  最大11
	unsigned int c : 3;	//3个二进制位
};
int main() 
{
	//溢出:超过存储范围
	struct Flag flag;
	flag.a = 5;
	flag.b = 1;
	flag.c = 8;  //_ _ _ _  1000
	printf("溢出:%u\n", flag.c);//无符号打印%u,打印后三位就是0
	printf("a=%u,b=%u\n", flag.a, flag.b);//打印5,1
	flag.c = 9;	//1001
	printf("溢出:%u\n", flag.c);//打印1
	return 0;
}

5 枚举类型和共用体

#include <stdio.h>
#include <string.h>

1.创建枚举
//enum 标识符{符号...}符号常量
enum Color{Red,Blue,Yellow,Green};
//默认值  Red=0 Blue=1 Yellow=2 Green=3

-----------------------------------------------------------------------------------------

2干啥用,咋用 
//程序中需要大量具有相同含义数字——>推箱子
2.1枚举菜单项
enum Menu {Insert,Delete,Modify,Sort,Search};
void keyDown()
{
	int key;
	scanf_s("%d", &key);
	switch (key) 
	{
	case Insert:
		break;
	case Delete:
		break;
	case Modify:
		break;
	case Sort:
		break;
	case Search:
		break;
	}
}
2.2初始化规则——>没有初始化的是前面的加1
enum Dir {Left=75,Right,Up=80,Down,Mid};
//Right: 75+1
//Down: 80+1
//Mid:Down+1=80+2

-----------------------------------------------------------------------------------------

3.C语言枚举类型当函数参数,也是可以传入整形变量
void print_value(enum Dir a) 
{
	printf("%d\n", a);
}

-----------------------------------------------------------------------------------------

共用体 union
union  Value
{
	char name[20];
	int age;
};

void test_union()
{
	//共同体 -->也叫联合体
	//所有变量共用同一块内存,一段内存给两个变量用
	//union Value value = { "A",65};  //不能同时初始化两个变量
	union Value value;
	strcpy_s(value.name, 20, "A");
	puts(value.name);
	printf("%d\n", value.age);	//65
	//value.age = 65;
	//printf("%d\n", value.age);
	//puts(value.name);
}

-----------------------------------------------------------------------------------------

int main() 
{	
    //1
	printf("Red:%d\n", Red);//1
	printf("Blue:%d\n", Blue);//2
    //2
	printf("Mid:%d\n", Mid);//82
	//3
	print_value(999);
	print_value(Left);
    //共用体
    test_union();
	return 0;
}

6 结构体嵌套

#include <stdio.h>
#include <stdlib.h>
//推荐嵌套 
struct Score 
{
	int math;
	int english;
	int py;
};
struct Info 
{
	char name[20];
	int age;
	char addr[20];
	char tel[14];
};
struct Student 
{
	struct Info info;
	struct Score score;
};

-----------------------------------------------------------------------------------------
    
//直接包含(个人感觉有一点别扭)
struct A 
{
	int a;
	struct B 
	{
		int age;
		int num;
	}b;//一般这里跟一个变量
};

-----------------------------------------------------------------------------------------

int main() 
{
	struct Student stu = { {"小芳",18,"火星","18934873456"},{88,89,99}};
	printf("%s\t%d\t%s\t%s\t%d\t%d\t%d\n", 
		stu.info.name,
		stu.info.age,
		stu.info.addr,
		stu.info.tel,
		stu.score.math,
		stu.score.english,
		stu.score.py);

	struct A object = { 1,{23,45} };
	printf("%d\t%d\t%d\n", object.a, object.b.age, object.b.num);
	return 0;
}

7 结构体中的指针

时刻考虑内存问题

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <assert.h>
struct MM 
{
	char* name;
	int age;
};


//返回一个堆区的结构体变量
struct MM* create_mm(const char* name, int age) 
{
	struct MM* temp = (struct MM*)malloc(sizeof(struct MM));
	assert(temp);
	//temp->name = name;
	int length = strlen(name) + 1;
	temp->name = (char*)malloc(length);
	strcpy_s(temp->name, length, name);
	temp->age = age;
	return temp;
}

-----------------------------------------------------------------------------------------

int main() 
{
	struct MM one = { "ILoveyou",18 };
	struct MM two = { "小花",28 };
	struct MM temp = one;
	one = two;
	two = temp;
	printf("%s\t%d\n", one.name, one.age);
	printf("%s\t%d\n", two.name, two.age);
	//这个操作就是错误的
	// name是指针,没有内存
	//strcpy_s(one.name, 20, "小可爱");
	one.name = (char*)malloc(20);
	assert(one.name);
	strcpy_s(one.name, 20, "小可爱");
	printf("%s\t%d\n", one.name, one.age);
	struct MM* p = create_mm("小宝贝", 18);
	printf("%s\t%d\n", p->name, p->age);
	return 0;

8 结构体内存问题

(有学业需求的)封装——>减少内存,传输数据就少点

#include <stdio.h>
#include <string.h>
#include <malloc.h>
#include <assert.h>
1
//具体占用内存多少其实不是那么重要,
//怎么写才省内存才是重要的。
    
-----------------------------------------------------------------------------------------
    
2 内存对齐
(按成员最长字节对齐)
//char 可以拆分
//char可以和int以及float填补
struct MM 
{	
	char name[20];  //8+8+ 多4可以和下面的补一起
	int age;        //4+上面的多4
	double num;		//8
    //8+8+4+4+8=32
};

struct Data 
{
	char name[20];  //8+8 多4不能和下面补,要多补4
	double num;     //8
	int age;        //4+补4
    //8+8+4+4+8+4+4=40
};

一样的东西,写法顺序不一样,内存多占8位,数据特别大的时候流量就用的多了
-----------------------------------------------------------------------------------------

struct Value 
{
	char name[3];	//3 补1
	int age;        //4
	char id[2];     //2 补2
};
struct Test 
{	//64位中,指针占8个字节
	char name[9];   //8+1+补7
	char* p;		//8
	int age;        //4 +补4
    //8+1+7+8+4+4=20
};

-----------------------------------------------------------------------------------------
int main() 
{
    //1
	//自己写代码一个sizeof就行了,自己不用算内存是多少
	printf("%zd\n", sizeof(struct MM));
	struct MM* p = (struct MM*)malloc(sizeof(struct MM));
	assert(p);
	
    //2
	printf("MM:%zd\n", sizeof(struct MM));
	printf("Data:%zd\n", sizeof(struct Data));
	printf("Value:%zd\n", sizeof(struct Value));
	printf("Test:%zd\n", sizeof(struct Test));
	return 0;
}



#pragma pack(8)写在头文件中——>代表着指定每个成员按照8字节对齐
;