工资管理系统(学校期末作业)

news2024/11/16 18:32:10

一、 题目要求

1、需求分析

工资信息存放在文件中,提供文件的输入、输出等操作;要浏览,提供显示、排序操作;查询功能要求实现查找操作;提供键盘式选择菜单以实现功能选择。

2、总体设计

系统可分为信息输入、信息添加、信息浏览、信息排序、信息查询和信息统计模块。

3、详细设计

工资信息采用结构体数组:

Struct Salary_Info
{ 
   int Card_No;                   //工资卡号
   Char name[20];                 //姓名
   int month;                     //月份
   float Init_Salary;               //应发工资 
   float Water_Rate;               //水费
   float Electric_Rate;             //电费
   float tax;                     //税金
   float Final_Salary;             //实发工资
};

(1)主函数提供输入、处理和输出部分的函数调用,各功能模块采用菜单方式选择。

(2)输入模块
按照工资卡号、姓名、月份、应发工资、水费、电费的顺序输入信息,税金和实发工资根据输入的信息进行计算得到,这些信息被录入到文件中。
文件操作函数:fopen,fwrite,fclose
税金的计算:
if (应发工资<=5000)
税金=0;
else if (应发工资>5000&&应发工资<=8000)
税金 =(应发工资-5000) * 5%;
else if (应发工资>8000)
税金 =(应发工资-8000) * 10%;
实发工资 = 应发工资 - 水费 - 电费 - 税金。

(3)浏览模块
分屏显示职工工资信息,可以指定10个1屏,按任意键显示下一屏。通过菜单选择按照工资卡号还是姓名浏览。如果按照卡号浏览,则显示的记录按照卡号升序输出;按照姓名浏览则按照字典序输出(调用排序模块的排序功能)。

(4)排序模块
排序模块提供菜单选择,实现按照工资卡号升序、实发工资降序以及姓名字典序排序。排序方法可以选择冒泡排序、插入排序、选择排序等。

(5)查询模块
实现按照工资卡号和姓名的查询,采用基本的查找方法即可。

(6)统计模块
输入起止月份,按照职工卡号把起止月份之间的实发工资金额累加。


二、设计思路

1.结构体的设置

在员工信息的基础上再封装一层结构体,将整个系统的当前存量和最大容量放入此结构体中。

typedef struct Salary_Info // 职工工资信息
{
	int Card_No;
	char name[20];
	int month;
	float Init_Salary;
	float Water_Rate;
	float Electric_Rate;
	float tax;
	float Final_Salary;
}Salary_Info;

typedef struct System // 进行封装
{
	Salary_Info* Data; // 指向职工工资信息数组的指针
	int size; // 记载当前存量
	int capacity; // 记载当前最大容量
}System;

2.储存信息的数据结构

利用可倍增的动态开辟的数组来储存所有的工资信息。
并利用定义的Check_Capacity()函数来实现扩容和初始化容量。

void Check_Capacity(System* ptr) // 检测容量与扩容
{
	if (ptr->size == ptr->capacity)
	{
		int New_Capacity = ptr->capacity ? ptr->capacity * 2 : DEFAULT; // 每次容量不足扩增两倍, 若容量为0则赋予默认容量
		Salary_Info* tmp = (Salary_Info*)realloc(ptr->Data, New_Capacity * sizeof(Salary_Info)); // 采用临时指针来接收,避免realloc失败导致的内存泄漏问题
		if (!tmp)
		{
			perror("Check_Capacity");
			exit(-1);
		}
		ptr->capacity = New_Capacity;
		ptr->Data = tmp;
	}
}

3.初始化系统以及录入数据

void LoadInfo(System* ptr) // 录入数据
{
	FILE* pf = fopen("Salary_Info.dat", "rb");
	if (!pf)
	{
		perror("LoadInfo");
		return;
	}
	Salary_Info tmp; // 创建临时变量来接收读取自文件的数据
	while (fread(&tmp, sizeof(Salary_Info), 1, pf))
	{
		Check_Capacity(ptr); // 检测容量,同时进行扩增
		memcpy(&ptr->Data[ptr->size], &tmp, sizeof(Salary_Info)); // 将临时变量的数据拷贝至Data中
		ptr->size++;
	}
	fclose(pf);
}
void Init_System(System* ptr) // 初始化系统
{
	ptr->capacity = ptr->size = 0;
	ptr->Data = NULL;
	LoadInfo(ptr);
}

4.添加职工工资信息

考虑到快速添加和单个添加,将整个添加板块分为了单次添加和批量添加。

void Single_Add(System* ptr) // 单次输入职工工资信息
{
	// 检测容量,若容量不足则扩容
	Check_Capacity(ptr);

	// 格式化输入
	printf("\n\n\n\n工资卡号(10位):>");
	scanf("%d", &ptr->Data[ptr->size].Card_No);
	printf("\n\n姓名:>");
	scanf("%s", &ptr->Data[ptr->size].name);
	printf("\n\n月份:>");
	scanf("%d", &ptr->Data[ptr->size].month);
	printf("\n\n应发工资:>");
	scanf("%f", &ptr->Data[ptr->size].Init_Salary);
	printf("\n\n水费:>");
	scanf("%f", &ptr->Data[ptr->size].Water_Rate);
	printf("\n\n电费:>");
	scanf("%f", &ptr->Data[ptr->size].Electric_Rate);

	// 计算税金以及其实发工资
	if (ptr->Data[ptr->size].Init_Salary <= 5000.0)
		ptr->Data[ptr->size].tax = 0.0;
	else if (ptr->Data[ptr->size].Init_Salary > 5000 && ptr->Data[ptr->size].Init_Salary <= 8000)
		ptr->Data[ptr->size].tax = (ptr->Data[ptr->size].Init_Salary - 5000.0) * 0.05;
	else if (ptr->Data[ptr->size].Init_Salary > 8000)
		ptr->Data[ptr->size].tax = (ptr->Data[ptr->size].Init_Salary - 8000.0) * 0.1;
	ptr->Data[ptr->size].Final_Salary = ptr->Data[ptr->size].Init_Salary - ptr->Data[ptr->size].Water_Rate - ptr->Data[ptr->size].Electric_Rate - ptr->Data[ptr->size].tax;
	ptr->size++;
}
void Batch(System* ptr) // 批量输入职工工资信息
{
	// 检测容量,若容量不足则扩容
	Check_Capacity(ptr);

	// 格式化输入
	scanf("%d", &ptr->Data[ptr->size].Card_No);
	scanf("%s", &ptr->Data[ptr->size].name);
	scanf("%d", &ptr->Data[ptr->size].month);
	scanf("%f", &ptr->Data[ptr->size].Init_Salary);
	scanf("%f", &ptr->Data[ptr->size].Water_Rate);
	scanf("%f", &ptr->Data[ptr->size].Electric_Rate);

	// 计算税金以及其实发工资
	if (ptr->Data[ptr->size].Init_Salary <= 5000.0)
		ptr->Data[ptr->size].tax = 0.0;
	else if (ptr->Data[ptr->size].Init_Salary > 5000 && ptr->Data[ptr->size].Init_Salary <= 8000)
		ptr->Data[ptr->size].tax = (ptr->Data[ptr->size].Init_Salary - 5000.0) * 0.05;
	else if (ptr->Data[ptr->size].Init_Salary > 8000)
		ptr->Data[ptr->size].tax = (ptr->Data[ptr->size].Init_Salary - 8000.0) * 0.1;
	ptr->Data[ptr->size].Final_Salary = ptr->Data[ptr->size].Init_Salary - ptr->Data[ptr->size].Water_Rate - ptr->Data[ptr->size].Electric_Rate - ptr->Data[ptr->size].tax;
	ptr->size++;
}
void Add_System(System* ptr) // 添加职工工资信息系统
{
	system("cls");
	Check_Capacity(ptr);
	printf("\n\n\t\t\t\t\t\t1.单次添加\n");
	printf("\n\n\t\t\t\t\t\t2.批量添加\n");
	printf("\n\n\t\t\t\t\t\t0.返回上一级菜单\n");
	printf("\n\t\t\t\t\t\t请选择:> ");
	int op;
	scanf("%d", &op);
	system("cls");
	switch (op)
	{
	case 0:
		break;
	case 1:
		Single_Add(ptr);
		system("cls");
		printf("\n\n\n\n\n\n\n\t\t\t\t\t\t添加成功!(已存在信息:> %d)", ptr->size);
		printf("\n\n按回车键返回!");
		getchar();
		while (!getchar());
		break;
	case 2:
		printf("\n\n\n\n\n\n\n\t\t\t\t\t\t请输入批量添加的次数:> ");
		int n;
		scanf("%d", &n);
		system("cls");
		printf("按照工资卡号(10位)、姓名、月份、应发工资、水费、电费的顺序输入信息(x%d):\n", n);
		while (n--) Batch(ptr);
		system("cls");
		printf("\n\n\n\n\n\n\n\t\t\t\t\t\t添加成功!(已存在的信息:> %d)\n", ptr->size);
		printf("\n\n按回车键返回!");
		getchar();
		while (!getchar());
		break;
	default:
		printf("请正确输入!\n");
		printf("\n\n按回车键返回!");
		getchar();
		while (!getchar());
		break;
	}
}

5.自实现快速排序

自实现swap()函数

由于c语言无内置的swap()交换函数,需要自己手动实现一个。

void swap(Salary_Info* e1, Salary_Info* e2) // 用于交换职工工资信息
{
	Salary_Info tmp = *e1;
	*e1 = *e2;
	*e2 = tmp;
}

实现工资卡号升序的快速排序

void quick_sort_Card_No_greater(System* ptr, int l, int r) // 按照工资卡号升序的快速排序
{
	if (l >= r) return;
	int x = ptr->Data[(l + r) >> 1].Card_No;
	int i = l - 1, j = r + 1;
	while (i < j)
	{
		do i++; while (ptr->Data[i].Card_No < x);
		do j--; while (ptr->Data[j].Card_No > x);
		if (i < j) swap(&ptr->Data[i], &ptr->Data[j]);
	}
	quick_sort_Card_No_greater(ptr, l, j);
	quick_sort_Card_No_greater(ptr, j + 1, r);
}

实现实发工资降序的快速排序

void quick_sort_Final_Salary_less(System* ptr, int l, int r) // 按照实发工资降序的快速排序
{
	if (l >= r) return;
	float x = ptr->Data[(l + r) >> 1].Final_Salary;
	int i = l - 1, j = r + 1;
	while (i < j)
	{
		do i++; while (ptr->Data[i].Final_Salary > x);
		do j--; while (ptr->Data[j].Final_Salary < x);
		if (i < j) swap(&ptr->Data[i], &ptr->Data[j]);
	}
	quick_sort_Final_Salary_less(ptr, l, j);
	quick_sort_Final_Salary_less(ptr, j + 1, r);
}

实现按姓名字典序的快速排序

void quick_sort_name(System* ptr, int l, int r) // 按照姓名字典序的快速排序
{
	if (l >= r) return;
	char* x = ptr->Data[(l + r) >> 1].name;
	int i = l - 1, j = r + 1;
	while (i < j)
	{
		do i++; while (strcmp(ptr->Data[i].name, x) < 0);
		do j--; while (strcmp(ptr->Data[j].name, x) > 0);
		if (i < j) swap(&ptr->Data[i], &ptr->Data[j]);
	}
	quick_sort_name(ptr, l, j);
	quick_sort_name(ptr, j + 1, r);
}

实现按照月份的快速排序

void quick_sort_Month_greater(System* ptr, int l, int r) // 按照工资月份升序的快速排序
{
	if (l >= r) return;
	int x = ptr->Data[(l + r) >> 1].month;
	int i = l - 1, j = r + 1;
	while (i < j)
	{
		do i++; while (ptr->Data[i].month < x);
		do j--; while (ptr->Data[j].month > x);
		if (i < j) swap(&ptr->Data[i], &ptr->Data[j]);
	}
	quick_sort_Month_greater(ptr, l, j);
	quick_sort_Month_greater(ptr, j + 1, r);
}

6.浏览工资信息(一屏十条可上下页翻找)(同时进行修改删除)

效果图:
效果图

void Scan_System(System* ptr)
{
	if (!ptr->size)
	{
		system("cls");
		printf("\n\n\n\n\t\t\t\t\t\t当前未有任何信息!\n");
		printf("\n\n按回车键返回主菜单!");
		getchar();
		while (!getchar());
		return;
	}
	int op, i = 0, n = 0;
	system("cls");
	printf("\n\n\t\t\t\t\t\t1.按照工资卡号升序浏览\n");
	printf("\n\n\t\t\t\t\t\t2.按照实发工资降序浏览\n");
	printf("\n\n\t\t\t\t\t\t3.按照姓名字典序浏览\n");
	printf("\n\n\t\t\t\t\t\t请选择:> ");
	scanf("%d", &op);
	switch (op)
	{
	case 1:
		quick_sort_Card_No_greater(ptr, 0, ptr->size - 1);
		break;
	case 2:
		quick_sort_Final_Salary_less(ptr, 0, ptr->size - 1);
		break;
	case 3:
		quick_sort_name(ptr, 0, ptr->size - 1);
		break;
	}
	do
	{
		system("cls");
		printf("%-4s\t%-10s\t%-8s\t%-2s\t%8s\t%4s\t%4s\t%8s\t%8s\n", "序号", "工资卡号", "姓名", "月份", "应发工资", "水费", "电费", "税金", "实发工资");
		for (int j = 0; j + i < ptr->size && j < 10; j++)
		{
			printf("%-4d\t%-10d\t%-8s\t%02d\t%8.2f\t%4.2f\t%4.2f\t%8.2f\t%8.2f\n",
				i + j + 1,
				ptr->Data[i + j].Card_No,
				ptr->Data[i + j].name,
				ptr->Data[i + j].month,
				ptr->Data[i + j].Init_Salary,
				ptr->Data[i + j].Water_Rate,
				ptr->Data[i + j].Electric_Rate,
				ptr->Data[i + j].tax,
				ptr->Data[i + j].Final_Salary
			);
		}
		printf("\n\t\t\t\t\t\t当前页数:(%d/%d)\n", i / 10 + 1, (ptr->size - 1) / 10 + 1);
		printf("-----------------------------------------------------------------------------------------------------------------------\n");
		if (i + 10 < ptr->size && i)
			printf("选项: 0.返回主菜单 \t 1.下一页 \t 2.上一页 \t 3.修改信息 \t 4.删除信息\n请选择:> ");
		else if (i + 10 < ptr->size && !i)
			printf("选项: 0.返回主菜单 \t 1.下一页 \t 3.修改信息 \t 4.删除信息\n请选择:> ");
		else if (i + 10 >= ptr->size && i)
			printf("选项: 0.返回主菜单 \t 2.上一页 \t 3.修改信息 \t 4.删除信息\n请选择:> ");
		else if (i + 10 >= ptr->size && !i)
			printf("选项: 0.返回主菜单 \t 3.修改信息 \t 4.删除信息\n请选择:> ");
		scanf("%d", &op);
		switch (op)
		{
		case 0:
			break;
		case 1:
			if (i + 10 < ptr->size) i += 10;
			break;
		case 2:
			if (i) i -= 10;
			break;
		case 3:
			printf("请输入对应的序号:> ");
			scanf("%d", &n);
			system("cls");
			printf("------------------------------------------------------修改目标------------------------------------------------------\n");
			printf("%-10s\t%-8s\t%2s\t%8s\t%4s\t%4s\t%8s\t%8s\n", "工资卡号", "姓名", "月份", "应发工资", "水费", "电费", "税金", "实发工资");
			printf("%-10d\t%-8s\t%02d\t%8.2f\t%4.2f\t%4.2f\t%8.2f\t%8.2f\n",
				ptr->Data[n - 1].Card_No,
				ptr->Data[n - 1].name,
				ptr->Data[n - 1].month,
				ptr->Data[n - 1].Init_Salary,
				ptr->Data[n - 1].Water_Rate,
				ptr->Data[n - 1].Electric_Rate,
				ptr->Data[n - 1].tax,
				ptr->Data[n - 1].Final_Salary
			);
			printf("-----------------------------------------------------------------------------------------------------------------------\n");

			printf("按照工资卡号(10位)、姓名、月份、应发工资、水费、电费的顺序输入信息(注意不要输入税金和实发工资):\n");
			// 修改对应目标
			printf("\n\n工资卡号(10位):>");
			scanf("%d", &ptr->Data[n - 1].Card_No);
			printf("\n\n姓名:>");
			scanf("%s", &ptr->Data[n - 1].name);
			printf("\n\n月份:>");
			scanf("%d", &ptr->Data[n - 1].month);
			printf("\n\n应发工资:>");
			scanf("%f", &ptr->Data[n - 1].Init_Salary);
			printf("\n\n水费:>");
			scanf("%f", &ptr->Data[n - 1].Water_Rate);
			printf("\n\n电费:>");
			scanf("%f", &ptr->Data[n - 1].Electric_Rate);

			// 更新修改之后的税金和实发工资
			if (ptr->Data[n - 1].Init_Salary <= 5000.0)
				ptr->Data[n - 1].tax = 0.0;
			else if (ptr->Data[n - 1].Init_Salary > 5000 && ptr->Data[n - 1].Init_Salary <= 8000)
				ptr->Data[n - 1].tax = (ptr->Data[n - 1].Init_Salary - 5000.0) * 0.05;
			else if (ptr->Data[n - 1].Init_Salary > 8000)
				ptr->Data[n - 1].tax = (ptr->Data[n - 1].Init_Salary - 8000.0) * 0.1;
			ptr->Data[n - 1].Final_Salary = ptr->Data[n - 1].Init_Salary - ptr->Data[n - 1].Water_Rate - ptr->Data[n - 1].Electric_Rate - ptr->Data[n - 1].tax;
			system("cls");
			printf("\n\n\n\n\n\n\n\t\t\t\t\t\t修改成功!\n");
			printf("\n\n按回车键返回主菜单!");
			getchar();
			while (!getchar());
			return;
			break;
		case 4:
			printf("请输入对应的序号:> ");
			scanf("%d", &n);
			for (int i = n - 1; i < ptr->size - 1; ++i) // 依次将后面数据往前移一位,弥补删除的位置
				swap(&ptr->Data[i], &ptr->Data[i + 1]);
			ptr->size--;
			system("cls");
			printf("\n\n\n\n\n\n\n\t\t\t\t\t\t删除成功!(已存在的信息: > %d)\n", ptr->size);
			printf("\n\n按回车键返回主菜单!");
			getchar();
			while (!getchar());
			return;
			break;
		}
	} while (op);
}

7.左右边界查询工资信息

对查询工资卡号和姓名,可以先进行对应的排序,然后再使用二分法查找。由于职工工
资信息里会出现多个重复的姓名与工资卡号,所以采用二分法查询符合条件的目标的左
右边界,再对处在这个边界范围中的所有目标依次打印。


左边界二分法查询工资卡号

int Find_Card_No_Left(System* ptr, int target) // 左边界二分法找出第一次出现目标的坐标
{
	int l = 0, r = ptr->size - 1;
	while (l < r) // 采用左边界二分法
	{
		int mid = (l + r) / 2;
		if (ptr->Data[mid].Card_No >= target) r = mid;
		else l = mid + 1;
	}
	return l;
}

右边界二分法查询工资卡号

int Find_Card_No_Right(System* ptr, int target) // 右边界二分法找出最后一次出现目标的坐标
{
	int l = 0, r = ptr->size - 1;
	while (l < r) // 采用右边界二分法
	{
		int mid = (l + r + 1) / 2;
		if (ptr->Data[mid].Card_No <= target) l = mid;
		else r = mid - 1;
	}
	return l;
}

左边界二分法查询姓名

int Find_Name_Left(System* ptr, char* name) // 左边界二分法找出第一次出现目标的坐标
{
	int l = 0, r = ptr->size - 1;
	while (l < r) // 采用左边界二分法来查找名字
	{
		int mid = (l + r) / 2;
		if (strcmp(ptr->Data[mid].name, name) >= 0) r = mid;
		else l = mid + 1;
	}
	return l;
}

右边界二分法查询姓名

int Find_Name_Right(System* ptr, char* name) // 右边界二分法找出最后一次出现目标的坐标
{
	int l = 0, r = ptr->size - 1;
	while (l < r) // 采用右边界二分法来查找名字
	{
		int mid = (l + r + 1) / 2;
		if (strcmp(ptr->Data[mid].name, name) <= 0) l = mid;
		else r = mid - 1;
	}
	return l;
}

整个查询系统框架的设置

void Query_System(System* ptr) // 查询职工工资信息
{
	int op;
	do
	{
		int begin, end;
		system("cls");
		printf("\n\n\t\t\t\t\t\t1.按照工资卡号的查询\n");
		printf("\n\n\t\t\t\t\t\t2.按照姓名的查询\n");
		printf("\n\n\t\t\t\t\t\t0.返回主菜单\n");
		printf("\n\n\t\t\t\t\t\t请选择:> ");
		scanf("%d", &op);
		switch (op)
		{
		case 0:
			break;
		case 1:
			system("cls");
			quick_sort_Card_No_greater(ptr, 0, ptr->size - 1); // 先按工资卡号进行一次排序,便于后面使用二分法查询
			printf("查询的工资卡号:> ");
			int target;
			scanf("%d", &target);
			begin = Find_Card_No_Left(ptr, target);
			end = Find_Card_No_Right(ptr, target);
			printf("查询结果:\n");
			if (ptr->Data[begin].Card_No != target) // 检测二分法求出的目标是否符合要求
			{
				printf("\n\n\n\n\n\n\n\t\t\t\t\t\t暂无此工资卡号!\n");
				printf("\n\n按回车键返回上一级菜单!");
				getchar();
				while (!getchar());
			}
			else
			{
				// 通过二分法得到的左右边界将所有符合要求的目标都打印出来
				printf("-----------------------------------------------------------------------------------------------------------------------\n");
				printf("%-10s\t%-8s\t%-2s\t%8s\t%-4s\t%-4s\t%8s\t%8s\n", "工资卡号", "姓名", "月份", "应发工资", "水费", "电费", "税金", "实发工资");
				for (int i = begin; i <= end; i++)
					printf("%-10d\t%-8s\t%02d\t%8.2f\t%-4.2f\t%-4.2f\t%8.2f\t%8.2f\n",
						ptr->Data[i].Card_No,
						ptr->Data[i].name,
						ptr->Data[i].month,
						ptr->Data[i].Init_Salary,
						ptr->Data[i].Water_Rate,
						ptr->Data[i].Electric_Rate,
						ptr->Data[i].tax,
						ptr->Data[i].Final_Salary
					);
				printf("-----------------------------------------------------------------------------------------------------------------------\n");
				printf("\n\n按回车键返回上一级菜单!");
				getchar();
				while (!getchar());
			}
			break;
		case 2:
			system("cls");
			quick_sort_name(ptr, 0, ptr->size - 1); // 先按姓名字典序进行一次排序,便于后面使用二分法查询
			printf("查询的姓名:> ");
			char name[20];
			scanf("%s", name);
			begin = Find_Name_Left(ptr, name);
			end = Find_Name_Right(ptr, name);
			printf("查询结果:\n");
			if (strcmp(ptr->Data[begin].name, name)) // 检测二分法求出的目标是否符合要求
			{
				printf("\n\n\n\n\n\n\n\t\t\t\t\t\t暂无此人!\n");
				printf("\n\n按回车键返回上一级菜单!");
				getchar();
				while (!getchar());
			}
			else
			{
				// 通过二分法得到的左右边界将所有符合要求的目标都打印出来
				printf("-----------------------------------------------------------------------------------------------------------------------\n");
				printf("%-10s\t%-8s\t%-2s\t%8s\t%-4s\t%-4s\t%8s\t%8s\n", "工资卡号", "姓名", "月份", "应发工资", "水费", "电费", "税金", "实发工资");
				for (int i = begin; i <= end; i++)
					printf("%-10d\t%-8s\t%02d\t%8.2f\t%-4.2f\t%-4.2f\t%8.2f\t%8.2f\n",
						ptr->Data[i].Card_No,
						ptr->Data[i].name,
						ptr->Data[i].month,
						ptr->Data[i].Init_Salary,
						ptr->Data[i].Water_Rate,
						ptr->Data[i].Electric_Rate,
						ptr->Data[i].tax,
						ptr->Data[i].Final_Salary
					);
				printf("-----------------------------------------------------------------------------------------------------------------------\n");
				printf("\n\n按回车键返回上一级菜单!");
				getchar();
				while (!getchar());
			}
			break;
		}
	} while (op);
}

8.统计模块的设计

先进行按照月份的快速排序,再依次统计。

void Statistics_System(System* ptr) // 统计职工工资信息
{
	system("cls");
	if (!ptr->size)
	{
		printf("\n\n\n\n\t\t\t\t\t\t当前未有任何信息!\n");
		printf("\n\n按回车键返回主菜单!");
		getchar();
		while (!getchar());
		return;
	}
	quick_sort_Month_greater(ptr, 0, ptr->size - 1);
	printf("\n\n\n\n\t\t\t\t请输入起止月份(两个数字并用空格隔开):> ");
	int l, r, target;
	double sum = 0.0;
	scanf("%d%d", &l, &r);
	printf("\n\t\t\t\t请输入要查询的职工卡号(10位):> ");
	scanf("%d", &target);
	for (int i = 0; i < ptr->size; ++i) // 遍历所有符合的目标进行相加
	{
		if (ptr->Data[i].month >= l && ptr->Data[i].month <= r && ptr->Data[i].Card_No == target)
			sum += ptr->Data[i].Final_Salary;
	}
	printf("\n\t\t\t\t该职工%d~%d月份间实发工资总和为:> %.2lf\n", l, r, sum);
	printf("\n\n按回车键返回主菜单!");
	getchar();
	while (!getchar());
}

9.全部数据销毁

void Destroy_System(System* ptr) // 清除系统所有数据
{
	free(ptr->Data);
	ptr->Data = NULL;
	ptr->size = ptr->capacity = 0;
	system("cls");
	printf("\n\n\n\n\t\t\t\t\t\t清除系统所有数据成功!(记得保存!)\n");
	printf("\n\n按回车键返回主菜单!");
	getchar();
	while (!getchar());
}

10.退出并保存数据至文件

void Save_System(const System* ptr) // 保存数据,存放至文件中
{
	FILE* pf = fopen("Salary_Info.dat", "wb");
	if (!pf) // 若打开或创建dat文件失败,报错退出
	{
		perror("Save_System");
		exit(-1);
	}
	for (int i = 0; i < ptr->size; ++i) fwrite(ptr->Data + i, sizeof(Salary_Info), 1, pf); // 逐个读取存放进去
	if (!ptr->size) fwrite(" ", 0, 0, pf); // 若此时系统数据为空,则将文件赋空
}

三、总代码

statement.h(存放函数声明)

#define _CRT_SECURE_NO_WARNINGS // 解除VS2022安全警告
#include<stdio.h>
#include<string.h>
#include<stdlib.h>

#define DEFAULT 5 // 定义默认的容量大小

typedef struct Salary_Info // 职工工资信息
{
	int Card_No;
	char name[20];
	int month;
	float Init_Salary;
	float Water_Rate;
	float Electric_Rate;
	float tax;
	float Final_Salary;
}Salary_Info;

typedef struct System // 进行封装
{
	Salary_Info* Data; // 指向职工工资信息数组的指针
	int size; // 记载当前存量
	int capacity; // 记载当前最大容量
}System;

void Mainmenu(); // 主菜单

void LoadInfo(System* ptr); // 录入数据

void Init_System(System* ptr); // 初始化系统

void Batch(System* ptr); // 批量输入职工工资信息
void Add_System(System* ptr); // 添加职工工资信息
void Single_Add(System* ptr); // 单次输入职工工资信息

void Scan_System(System* ptr); // 浏览职工工资信息

void swap(Salary_Info* e1, Salary_Info* e2); // 用于交换职工工资信息
void quick_sort_Card_No_greater(System* ptr, int l, int r); // 按照工资卡号升序的快速排序
void quick_sort_Final_Salary_less(System* ptr, int l, int r); // 按照实发工资降序的快速排序
void quick_sort_name(System* ptr, int l, int r); // 按照姓名字典序的快速排序

int Find_Card_No_Left(System* ptr, int target); // 左边界二分法查询工资卡号并返回左边界坐标
int Find_Card_No_Right(System* ptr, int target); // 右边界二分法查询工资卡号并返回右边界坐标
int Find_Name_Left(System* ptr, char* name); // 左边界二分法查询姓名返回左边界坐标
int Find_Name_Right(System* ptr, char* name); // 右边界二分法查询姓名返回右边界坐标
void Query_System(System* ptr); // 查询职工工资信息

void quick_sort_Month_greater(System* ptr, int l, int r); // 按照工资月份升序的快速排序
void Statistics_System(System* ptr); // 统计职工工资信息

void Destroy_System(System* ptr); // 清除所有数据

void Save_System(const System* ptr); // 保存系统数据

function.c(存放函数定义)

#include"statement.h"

void Mainmenu() // 主菜单
{
	system("cls");
	printf("\n\n\t\t\t\t\t工资管理系统 v1.0(请输入符合事实的数据)\n");
	printf("\n\n\t\t\t\t\t\t1.添加职工工资信息\n");
	printf("\n\t\t\t\t\t\t2.浏览职工工资信息\n");
	printf("\n\t\t\t\t\t\t3.查询职工工资信息\n");
	printf("\n\t\t\t\t\t\t4.统计职工工资信息\n");
	printf("\n\t\t\t\t\t\t5.清除系统所有数据\n");
	printf("\n\t\t\t\t\t\t0.退出并保存\n");
}


void Check_Capacity(System* ptr) // 检测容量与扩容
{
	if (ptr->size == ptr->capacity)
	{
		int New_Capacity = ptr->capacity ? ptr->capacity * 2 : DEFAULT; // 每次容量不足扩增两倍, 若容量为0则赋予默认容量
		Salary_Info* tmp = (Salary_Info*)realloc(ptr->Data, New_Capacity * sizeof(Salary_Info)); // 采用临时指针来接收,避免realloc失败导致的内存泄漏问题
		if (!tmp)
		{
			perror("Check_Capacity");
			exit(-1);
		}
		ptr->capacity = New_Capacity;
		ptr->Data = tmp;
	}
}


void LoadInfo(System* ptr) // 录入数据
{
	FILE* pf = fopen("Salary_Info.dat", "rb");
	if (!pf)
	{
		perror("LoadInfo");
		return;
	}
	Salary_Info tmp; // 创建临时变量来接收读取自文件的数据
	while (fread(&tmp, sizeof(Salary_Info), 1, pf))
	{
		Check_Capacity(ptr); // 检测容量,同时进行扩增
		memcpy(&ptr->Data[ptr->size], &tmp, sizeof(Salary_Info)); // 将临时变量的数据拷贝至Data中
		ptr->size++;
	}
	fclose(pf);
}
void Init_System(System* ptr) // 初始化系统
{
	ptr->capacity = ptr->size = 0;
	ptr->Data = NULL;
	LoadInfo(ptr);
}

void Single_Add(System* ptr) // 单次输入职工工资信息
{
	// 检测容量,若容量不足则扩容
	Check_Capacity(ptr);

	// 格式化输入
	printf("\n\n\n\n工资卡号(10位):>");
	scanf("%d", &ptr->Data[ptr->size].Card_No);
	printf("\n\n姓名:>");
	scanf("%s", &ptr->Data[ptr->size].name);
	printf("\n\n月份:>");
	scanf("%d", &ptr->Data[ptr->size].month);
	printf("\n\n应发工资:>");
	scanf("%f", &ptr->Data[ptr->size].Init_Salary);
	printf("\n\n水费:>");
	scanf("%f", &ptr->Data[ptr->size].Water_Rate);
	printf("\n\n电费:>");
	scanf("%f", &ptr->Data[ptr->size].Electric_Rate);

	// 计算税金以及其实发工资
	if (ptr->Data[ptr->size].Init_Salary <= 5000.0)
		ptr->Data[ptr->size].tax = 0.0;
	else if (ptr->Data[ptr->size].Init_Salary > 5000 && ptr->Data[ptr->size].Init_Salary <= 8000)
		ptr->Data[ptr->size].tax = (ptr->Data[ptr->size].Init_Salary - 5000.0) * 0.05;
	else if (ptr->Data[ptr->size].Init_Salary > 8000)
		ptr->Data[ptr->size].tax = (ptr->Data[ptr->size].Init_Salary - 8000.0) * 0.1;
	ptr->Data[ptr->size].Final_Salary = ptr->Data[ptr->size].Init_Salary - ptr->Data[ptr->size].Water_Rate - ptr->Data[ptr->size].Electric_Rate - ptr->Data[ptr->size].tax;
	ptr->size++;
}
void Batch(System* ptr) // 批量输入职工工资信息
{
	// 检测容量,若容量不足则扩容
	Check_Capacity(ptr);

	// 格式化输入
	scanf("%d", &ptr->Data[ptr->size].Card_No);
	scanf("%s", &ptr->Data[ptr->size].name);
	scanf("%d", &ptr->Data[ptr->size].month);
	scanf("%f", &ptr->Data[ptr->size].Init_Salary);
	scanf("%f", &ptr->Data[ptr->size].Water_Rate);
	scanf("%f", &ptr->Data[ptr->size].Electric_Rate);

	// 计算税金以及其实发工资
	if (ptr->Data[ptr->size].Init_Salary <= 5000.0)
		ptr->Data[ptr->size].tax = 0.0;
	else if (ptr->Data[ptr->size].Init_Salary > 5000 && ptr->Data[ptr->size].Init_Salary <= 8000)
		ptr->Data[ptr->size].tax = (ptr->Data[ptr->size].Init_Salary - 5000.0) * 0.05;
	else if (ptr->Data[ptr->size].Init_Salary > 8000)
		ptr->Data[ptr->size].tax = (ptr->Data[ptr->size].Init_Salary - 8000.0) * 0.1;
	ptr->Data[ptr->size].Final_Salary = ptr->Data[ptr->size].Init_Salary - ptr->Data[ptr->size].Water_Rate - ptr->Data[ptr->size].Electric_Rate - ptr->Data[ptr->size].tax;
	ptr->size++;
}
void Add_System(System* ptr) // 添加职工工资信息系统
{
	system("cls");
	Check_Capacity(ptr);
	printf("\n\n\t\t\t\t\t\t1.单次添加\n");
	printf("\n\n\t\t\t\t\t\t2.批量添加\n");
	printf("\n\n\t\t\t\t\t\t0.返回上一级菜单\n");
	printf("\n\t\t\t\t\t\t请选择:> ");
	int op;
	scanf("%d", &op);
	system("cls");
	switch (op)
	{
	case 0:
		break;
	case 1:
		Single_Add(ptr);
		system("cls");
		printf("\n\n\n\n\n\n\n\t\t\t\t\t\t添加成功!(已存在信息:> %d)", ptr->size);
		printf("\n\n按回车键返回!");
		getchar();
		while (!getchar());
		break;
	case 2:
		printf("\n\n\n\n\n\n\n\t\t\t\t\t\t请输入批量添加的次数:> ");
		int n;
		scanf("%d", &n);
		system("cls");
		printf("按照工资卡号(10位)、姓名、月份、应发工资、水费、电费的顺序输入信息(x%d):\n", n);
		while (n--) Batch(ptr);
		system("cls");
		printf("\n\n\n\n\n\n\n\t\t\t\t\t\t添加成功!(已存在的信息:> %d)\n", ptr->size);
		printf("\n\n按回车键返回!");
		getchar();
		while (!getchar());
		break;
	default:
		printf("请正确输入!\n");
		printf("\n\n按回车键返回!");
		getchar();
		while (!getchar());
		break;
	}
}


void Scan_System(System* ptr)
{
	if (!ptr->size)
	{
		system("cls");
		printf("\n\n\n\n\t\t\t\t\t\t当前未有任何信息!\n");
		printf("\n\n按回车键返回主菜单!");
		getchar();
		while (!getchar());
		return;
	}
	int op, i = 0, n = 0;
	system("cls");
	printf("\n\n\t\t\t\t\t\t1.按照工资卡号升序浏览\n");
	printf("\n\n\t\t\t\t\t\t2.按照实发工资降序浏览\n");
	printf("\n\n\t\t\t\t\t\t3.按照姓名字典序浏览\n");
	printf("\n\n\t\t\t\t\t\t请选择:> ");
	scanf("%d", &op);
	switch (op)
	{
	case 1:
		quick_sort_Card_No_greater(ptr, 0, ptr->size - 1);
		break;
	case 2:
		quick_sort_Final_Salary_less(ptr, 0, ptr->size - 1);
		break;
	case 3:
		quick_sort_name(ptr, 0, ptr->size - 1);
		break;
	}
	do
	{
		system("cls");
		printf("%-4s\t%-10s\t%-8s\t%-2s\t%8s\t%4s\t%4s\t%8s\t%8s\n", "序号", "工资卡号", "姓名", "月份", "应发工资", "水费", "电费", "税金", "实发工资");
		for (int j = 0; j + i < ptr->size && j < 10; j++)
		{
			printf("%-4d\t%-10d\t%-8s\t%02d\t%8.2f\t%4.2f\t%4.2f\t%8.2f\t%8.2f\n",
				i + j + 1,
				ptr->Data[i + j].Card_No,
				ptr->Data[i + j].name,
				ptr->Data[i + j].month,
				ptr->Data[i + j].Init_Salary,
				ptr->Data[i + j].Water_Rate,
				ptr->Data[i + j].Electric_Rate,
				ptr->Data[i + j].tax,
				ptr->Data[i + j].Final_Salary
			);
		}
		printf("\n\t\t\t\t\t\t当前页数:(%d/%d)\n", i / 10 + 1, (ptr->size - 1) / 10 + 1);
		printf("-----------------------------------------------------------------------------------------------------------------------\n");
		if (i + 10 < ptr->size && i)
			printf("选项: 0.返回主菜单 \t 1.下一页 \t 2.上一页 \t 3.修改信息 \t 4.删除信息\n请选择:> ");
		else if (i + 10 < ptr->size && !i)
			printf("选项: 0.返回主菜单 \t 1.下一页 \t 3.修改信息 \t 4.删除信息\n请选择:> ");
		else if (i + 10 >= ptr->size && i)
			printf("选项: 0.返回主菜单 \t 2.上一页 \t 3.修改信息 \t 4.删除信息\n请选择:> ");
		else if (i + 10 >= ptr->size && !i)
			printf("选项: 0.返回主菜单 \t 3.修改信息 \t 4.删除信息\n请选择:> ");
		scanf("%d", &op);
		switch (op)
		{
		case 0:
			break;
		case 1:
			if (i + 10 < ptr->size) i += 10;
			break;
		case 2:
			if (i) i -= 10;
			break;
		case 3:
			printf("请输入对应的序号:> ");
			scanf("%d", &n);
			system("cls");
			printf("------------------------------------------------------修改目标------------------------------------------------------\n");
			printf("%-10s\t%-8s\t%2s\t%8s\t%4s\t%4s\t%8s\t%8s\n", "工资卡号", "姓名", "月份", "应发工资", "水费", "电费", "税金", "实发工资");
			printf("%-10d\t%-8s\t%02d\t%8.2f\t%4.2f\t%4.2f\t%8.2f\t%8.2f\n",
				ptr->Data[n - 1].Card_No,
				ptr->Data[n - 1].name,
				ptr->Data[n - 1].month,
				ptr->Data[n - 1].Init_Salary,
				ptr->Data[n - 1].Water_Rate,
				ptr->Data[n - 1].Electric_Rate,
				ptr->Data[n - 1].tax,
				ptr->Data[n - 1].Final_Salary
			);
			printf("-----------------------------------------------------------------------------------------------------------------------\n");

			printf("按照工资卡号(10位)、姓名、月份、应发工资、水费、电费的顺序输入信息(注意不要输入税金和实发工资):\n");
			// 修改对应目标
			printf("\n\n工资卡号(10位):>");
			scanf("%d", &ptr->Data[n - 1].Card_No);
			printf("\n\n姓名:>");
			scanf("%s", &ptr->Data[n - 1].name);
			printf("\n\n月份:>");
			scanf("%d", &ptr->Data[n - 1].month);
			printf("\n\n应发工资:>");
			scanf("%f", &ptr->Data[n - 1].Init_Salary);
			printf("\n\n水费:>");
			scanf("%f", &ptr->Data[n - 1].Water_Rate);
			printf("\n\n电费:>");
			scanf("%f", &ptr->Data[n - 1].Electric_Rate);

			// 更新修改之后的税金和实发工资
			if (ptr->Data[n - 1].Init_Salary <= 5000.0)
				ptr->Data[n - 1].tax = 0.0;
			else if (ptr->Data[n - 1].Init_Salary > 5000 && ptr->Data[n - 1].Init_Salary <= 8000)
				ptr->Data[n - 1].tax = (ptr->Data[n - 1].Init_Salary - 5000.0) * 0.05;
			else if (ptr->Data[n - 1].Init_Salary > 8000)
				ptr->Data[n - 1].tax = (ptr->Data[n - 1].Init_Salary - 8000.0) * 0.1;
			ptr->Data[n - 1].Final_Salary = ptr->Data[n - 1].Init_Salary - ptr->Data[n - 1].Water_Rate - ptr->Data[n - 1].Electric_Rate - ptr->Data[n - 1].tax;
			system("cls");
			printf("\n\n\n\n\n\n\n\t\t\t\t\t\t修改成功!\n");
			printf("\n\n按回车键返回主菜单!");
			getchar();
			while (!getchar());
			return;
			break;
		case 4:
			printf("请输入对应的序号:> ");
			scanf("%d", &n);
			for (int i = n - 1; i < ptr->size - 1; ++i) // 依次将后面数据往前移一位,弥补删除的位置
				swap(&ptr->Data[i], &ptr->Data[i + 1]);
			ptr->size--;
			system("cls");
			printf("\n\n\n\n\n\n\n\t\t\t\t\t\t删除成功!(已存在的信息: > %d)\n", ptr->size);
			printf("\n\n按回车键返回主菜单!");
			getchar();
			while (!getchar());
			return;
			break;
		}
	} while (op);
}


void swap(Salary_Info* e1, Salary_Info* e2) // 用于交换职工工资信息
{
	Salary_Info tmp = *e1;
	*e1 = *e2;
	*e2 = tmp;
}

void quick_sort_Card_No_greater(System* ptr, int l, int r) // 按照工资卡号升序的快速排序
{
	if (l >= r) return;
	int x = ptr->Data[(l + r) >> 1].Card_No;
	int i = l - 1, j = r + 1;
	while (i < j)
	{
		do i++; while (ptr->Data[i].Card_No < x);
		do j--; while (ptr->Data[j].Card_No > x);
		if (i < j) swap(&ptr->Data[i], &ptr->Data[j]);
	}
	quick_sort_Card_No_greater(ptr, l, j);
	quick_sort_Card_No_greater(ptr, j + 1, r);
}
void quick_sort_Final_Salary_less(System* ptr, int l, int r) // 按照实发工资降序的快速排序
{
	if (l >= r) return;
	float x = ptr->Data[(l + r) >> 1].Final_Salary;
	int i = l - 1, j = r + 1;
	while (i < j)
	{
		do i++; while (ptr->Data[i].Final_Salary > x);
		do j--; while (ptr->Data[j].Final_Salary < x);
		if (i < j) swap(&ptr->Data[i], &ptr->Data[j]);
	}
	quick_sort_Final_Salary_less(ptr, l, j);
	quick_sort_Final_Salary_less(ptr, j + 1, r);
}
void quick_sort_name(System* ptr, int l, int r) // 按照姓名字典序的快速排序
{
	if (l >= r) return;
	char* x = ptr->Data[(l + r) >> 1].name;
	int i = l - 1, j = r + 1;
	while (i < j)
	{
		do i++; while (strcmp(ptr->Data[i].name, x) < 0);
		do j--; while (strcmp(ptr->Data[j].name, x) > 0);
		if (i < j) swap(&ptr->Data[i], &ptr->Data[j]);
	}
	quick_sort_name(ptr, l, j);
	quick_sort_name(ptr, j + 1, r);
}


int Find_Card_No_Left(System* ptr, int target) // 左边界二分法找出第一次出现目标的坐标
{
	int l = 0, r = ptr->size - 1;
	while (l < r) // 采用左边界二分法
	{
		int mid = (l + r) / 2;
		if (ptr->Data[mid].Card_No >= target) r = mid;
		else l = mid + 1;
	}
	return l;
}
int Find_Card_No_Right(System* ptr, int target) // 右边界二分法找出最后一次出现目标的坐标
{
	int l = 0, r = ptr->size - 1;
	while (l < r) // 采用右边界二分法
	{
		int mid = (l + r + 1) / 2;
		if (ptr->Data[mid].Card_No <= target) l = mid;
		else r = mid - 1;
	}
	return l;
}
int Find_Name_Left(System* ptr, char* name) // 左边界二分法找出第一次出现目标的坐标
{
	int l = 0, r = ptr->size - 1;
	while (l < r) // 采用左边界二分法来查找名字
	{
		int mid = (l + r) / 2;
		if (strcmp(ptr->Data[mid].name, name) >= 0) r = mid;
		else l = mid + 1;
	}
	return l;
}
int Find_Name_Right(System* ptr, char* name) // 右边界二分法找出最后一次出现目标的坐标
{
	int l = 0, r = ptr->size - 1;
	while (l < r) // 采用右边界二分法来查找名字
	{
		int mid = (l + r + 1) / 2;
		if (strcmp(ptr->Data[mid].name, name) <= 0) l = mid;
		else r = mid - 1;
	}
	return l;
}
void Query_System(System* ptr) // 查询职工工资信息
{
	int op;
	do
	{
		int begin, end;
		system("cls");
		printf("\n\n\t\t\t\t\t\t1.按照工资卡号的查询\n");
		printf("\n\n\t\t\t\t\t\t2.按照姓名的查询\n");
		printf("\n\n\t\t\t\t\t\t0.返回主菜单\n");
		printf("\n\n\t\t\t\t\t\t请选择:> ");
		scanf("%d", &op);
		switch (op)
		{
		case 0:
			break;
		case 1:
			system("cls");
			quick_sort_Card_No_greater(ptr, 0, ptr->size - 1); // 先按工资卡号进行一次排序,便于后面使用二分法查询
			printf("查询的工资卡号:> ");
			int target;
			scanf("%d", &target);
			begin = Find_Card_No_Left(ptr, target);
			end = Find_Card_No_Right(ptr, target);
			printf("查询结果:\n");
			if (ptr->Data[begin].Card_No != target) // 检测二分法求出的目标是否符合要求
			{
				printf("\n\n\n\n\n\n\n\t\t\t\t\t\t暂无此工资卡号!\n");
				printf("\n\n按回车键返回上一级菜单!");
				getchar();
				while (!getchar());
			}
			else
			{
				// 通过二分法得到的左右边界将所有符合要求的目标都打印出来
				printf("-----------------------------------------------------------------------------------------------------------------------\n");
				printf("%-10s\t%-8s\t%-2s\t%8s\t%-4s\t%-4s\t%8s\t%8s\n", "工资卡号", "姓名", "月份", "应发工资", "水费", "电费", "税金", "实发工资");
				for (int i = begin; i <= end; i++)
					printf("%-10d\t%-8s\t%02d\t%8.2f\t%-4.2f\t%-4.2f\t%8.2f\t%8.2f\n",
						ptr->Data[i].Card_No,
						ptr->Data[i].name,
						ptr->Data[i].month,
						ptr->Data[i].Init_Salary,
						ptr->Data[i].Water_Rate,
						ptr->Data[i].Electric_Rate,
						ptr->Data[i].tax,
						ptr->Data[i].Final_Salary
					);
				printf("-----------------------------------------------------------------------------------------------------------------------\n");
				printf("\n\n按回车键返回上一级菜单!");
				getchar();
				while (!getchar());
			}
			break;
		case 2:
			system("cls");
			quick_sort_name(ptr, 0, ptr->size - 1); // 先按姓名字典序进行一次排序,便于后面使用二分法查询
			printf("查询的姓名:> ");
			char name[20];
			scanf("%s", name);
			begin = Find_Name_Left(ptr, name);
			end = Find_Name_Right(ptr, name);
			printf("查询结果:\n");
			if (strcmp(ptr->Data[begin].name, name)) // 检测二分法求出的目标是否符合要求
			{
				printf("\n\n\n\n\n\n\n\t\t\t\t\t\t暂无此人!\n");
				printf("\n\n按回车键返回上一级菜单!");
				getchar();
				while (!getchar());
			}
			else
			{
				// 通过二分法得到的左右边界将所有符合要求的目标都打印出来
				printf("-----------------------------------------------------------------------------------------------------------------------\n");
				printf("%-10s\t%-8s\t%-2s\t%8s\t%-4s\t%-4s\t%8s\t%8s\n", "工资卡号", "姓名", "月份", "应发工资", "水费", "电费", "税金", "实发工资");
				for (int i = begin; i <= end; i++)
					printf("%-10d\t%-8s\t%02d\t%8.2f\t%-4.2f\t%-4.2f\t%8.2f\t%8.2f\n",
						ptr->Data[i].Card_No,
						ptr->Data[i].name,
						ptr->Data[i].month,
						ptr->Data[i].Init_Salary,
						ptr->Data[i].Water_Rate,
						ptr->Data[i].Electric_Rate,
						ptr->Data[i].tax,
						ptr->Data[i].Final_Salary
					);
				printf("-----------------------------------------------------------------------------------------------------------------------\n");
				printf("\n\n按回车键返回上一级菜单!");
				getchar();
				while (!getchar());
			}
			break;
		}
	} while (op);
}


void quick_sort_Month_greater(System* ptr, int l, int r) // 按照工资月份升序的快速排序
{
	if (l >= r) return;
	int x = ptr->Data[(l + r) >> 1].month;
	int i = l - 1, j = r + 1;
	while (i < j)
	{
		do i++; while (ptr->Data[i].month < x);
		do j--; while (ptr->Data[j].month > x);
		if (i < j) swap(&ptr->Data[i], &ptr->Data[j]);
	}
	quick_sort_Month_greater(ptr, l, j);
	quick_sort_Month_greater(ptr, j + 1, r);
}
void Statistics_System(System* ptr) // 统计职工工资信息
{
	system("cls");
	if (!ptr->size)
	{
		printf("\n\n\n\n\t\t\t\t\t\t当前未有任何信息!\n");
		printf("\n\n按回车键返回主菜单!");
		getchar();
		while (!getchar());
		return;
	}
	quick_sort_Month_greater(ptr, 0, ptr->size - 1);
	printf("\n\n\n\n\t\t\t\t请输入起止月份(两个数字并用空格隔开):> ");
	int l, r, target;
	double sum = 0.0;
	scanf("%d%d", &l, &r);
	printf("\n\t\t\t\t请输入要查询的职工卡号(10位):> ");
	scanf("%d", &target);
	for (int i = 0; i < ptr->size; ++i) // 遍历所有符合的目标进行相加
	{
		if (ptr->Data[i].month >= l && ptr->Data[i].month <= r && ptr->Data[i].Card_No == target)
			sum += ptr->Data[i].Final_Salary;
	}
	printf("\n\t\t\t\t该职工%d~%d月份间实发工资总和为:> %.2lf\n", l, r, sum);
	printf("\n\n按回车键返回主菜单!");
	getchar();
	while (!getchar());
}


void Destroy_System(System* ptr) // 清除系统所有数据
{
	free(ptr->Data);
	ptr->Data = NULL;
	ptr->size = ptr->capacity = 0;
	system("cls");
	printf("\n\n\n\n\t\t\t\t\t\t清除系统所有数据成功!(记得保存!)\n");
	printf("\n\n按回车键返回主菜单!");
	getchar();
	while (!getchar());
}


void Save_System(const System* ptr) // 保存数据,存放至文件中
{
	FILE* pf = fopen("Salary_Info.dat", "wb");
	if (!pf) // 若打开或创建dat文件失败,报错退出
	{
		perror("Save_System");
		exit(-1);
	}
	for (int i = 0; i < ptr->size; ++i) fwrite(ptr->Data + i, sizeof(Salary_Info), 1, pf); // 逐个读取存放进去
	if (!ptr->size) fwrite(" ", 0, 0, pf); // 若此时系统数据为空,则将文件赋空
}

main.c(存放主函数)

#include"statement.h"

int main()
{
	system("mode con: cols=120 lines=30"); // 维持窗口大小
	enum Option // 通过枚举常量将对应单词赋予值
	{
		Exit, // 退出并保存
		Add, // 添加职工工资信息
		Scan, // 浏览职工工资信息
		Query, // 查询职工工资信息
		Statistics, // 统计职工工资信息
		Destroy, // 清除系统所有数据
	};
	System s;
	Init_System(&s); // 对系统进行初始化
	int op;
	do
	{
		Mainmenu();
		printf("\n\t\t\t\t\t\t请选择:> ");
		scanf("%d", &op);
		switch (op)
		{
		case Exit:
			Save_System(&s); // 对系统进行保存
			free(s.Data); // 释放动态开辟的内存
			s.Data = NULL;
			break;
		case Add:
			Add_System(&s);
			break;
		case Scan:
			Scan_System(&s);
			break;
		case Query:
			Query_System(&s);
			break;
		case Statistics:
			Statistics_System(&s);
			break;
		case Destroy:
			Destroy_System(&s);
			break;
		default:
			system("cls");
			printf("\n\n\n\n\t\t\t\t\t\t请正确输入!\n");
			printf("\n\n按回车键返回!");
			getchar();
			while (!getchar());
			break;
		}
	} while (op);
	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/676835.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

洛谷 P2782 友好城市 排序 动态规划

题目描述 有一条横贯东西的大河&#xff0c;河有笔直的南北两岸&#xff0c;岸上各有位置各不相同的N个城市。北岸的每个城市有且仅有一个友好城市在南岸&#xff0c;而且不同城市的友好城市不相同。每对友好城市都向政府申请在河上开辟一条直线航道连接两个城市&#xff0c;但…

servlet 技能总结

Servlet介绍 Servlet是Server Applet的简称&#xff0c;称为服务端小程序&#xff0c;是JavaEE平台下的技术标准&#xff0c;基于Java语言编写的服务端程序。Web容器或应用服务器实现了Servlet标准所以Servlet需运行在Web容器或应用服务器中。Servlet主要功能在于能在服务器中执…

【前端技术】uni-app 01:快速开始

开个新坑&#xff0c;学习一下 uni-app&#xff0c;之后也想 uni-app 来做些事&#xff0c;虽然我主业是后端&#xff0c;但 uni-app 其作为一个高效生产力工具&#xff0c;个人认为非常有必要学习一下~ 目录 1 uni-app 介绍 1.1 uni-app 由来 1.2 为什么选择 uni-app 1.3 …

Win7 64位 VS2015及MinGW环境编译矢量库agg-2.5和cairo-1.14.6

书接上文&#xff0c;昨天装了MinGW&#xff0c;主要原因之一是要用到MSYS&#xff0c;所以顺手把FFMPEG又编译了一遍。 回到主题&#xff0c;其实我是想编译矢量库&#xff0c;因为最近要学习一些计算几何算法&#xff0c;所以找个方便的2D画图库就很重要。 说白了其实是懒得…

我把GPT 的学习轨迹可视化了竟和人类十分类似 |ACL2023

回想一下我们小时候是如何习得一门语言的&#xff1f;一般而言&#xff0c;在人类婴儿出生第一年内&#xff0c;最开始婴儿只能模仿式的说出一些“音素”&#xff0c;说出一些最简单与基本的单词或句子&#xff0c;而伴随着成长&#xff0c;在大约一岁到三岁的阶段&#xff0c;…

windows进程结构体

了解进程线程的概念后&#xff0c;我们就来看看windows里面的进程长什么样子的。进程本质上就是一个结构体。在Linux里面也称之为进程描述符。当操作系统创建一个进程的时候&#xff0c;它会填充一个结构体&#xff0c;往这个结构体里写入数据&#xff0c;这个结构体就用于管理…

Queue,List,Deque联系

如图所示&#xff0c;可以得出LinkedList既可以是双向链表也可以是双端队列&#xff0c;Deque接口继承了Queue接口 Queue add(E):boolean 在队尾添加元素&#xff0c;添加成功返回true&#xff0c;如果队列已满无法添加则抛出异常。offer(E):boolean 在队尾添加元素&#xff0…

linux mail -s发送邮件异常解决

异常&#xff1a; Error initializing NSS: Unknown error -8015. "/root/dead.letter" 11/301 . . . message not sent. 出现此问题&#xff0c;大概率是和证书相关。如果没有安装证书&#xff0c;请先安装&#xff1a; 1&#xff0c;下载 yum -y install mailx …

Python采集某xsp内容, m3u8视频内容下载

前言 大家早好、午好、晚好吖 ❤ ~欢迎光临本文章 环境使用: Python 3.8 Pycharm 专业版 模块使用: import requests >>> pip install requests import re 正则表达式 解析数据 import json 基本步骤去实现 一. 数据来源分析 通过开发者工具进行抓包分析, 分…

轻松构建交互式应用程序:探索Gradio Components模块的神奇世界!

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…

抽头延迟线信道模型

本专栏包含信息论与编码的核心知识&#xff0c;按知识点组织&#xff0c;可作为教学或学习的参考。markdown版本已归档至【Github仓库&#xff1a;https://github.com/timerring/information-theory 】或者公众号【AIShareLab】回复 信息论 获取。 文章目录 时变多径信道的信道…

突破技术边界,开创“粽“享未来

突破技术边界&#xff0c;开创“粽“享未来 端午节的由来端午节的习俗端午祈福 博主 默语带您 Go to New World. ✍ 个人主页—— 默语 的博客&#x1f466;&#x1f3fb; 《java 面试题大全》 &#x1f369;惟余辈才疏学浅&#xff0c;临摹之作或有不妥之处&#xff0c;还请读…

南京阿里云代理商:阿里云服务器的可扩展性和弹性如何?是否支持按需付费?

南京阿里云代理商&#xff1a;阿里云服务器的可扩展性和弹性如何&#xff1f;是否支持按需付费&#xff1f;   一、阿里云服务器的可扩展性   阿里云作为业界知名的云服务提供商&#xff0c;其服务器具有极强的可扩展性。可扩展性主要体现在以下几方面&#xff1a;   1. …

行为型模式--状态模式

目录 举例 状态模式 定义 结构 代码实现 优缺点 优点&#xff1a; 缺点&#xff1a; 使用场景 举例 【例】通过按钮来控制一个电梯的状态&#xff0c;一个电梯有开门状态&#xff0c;关门状态&#xff0c;停止状态&#xff0c;运行状态。每一 种状态改变&#xff0c;都…

Xdebug的安装及使用

Xdebug的安装及使用 前言一、Xdebug如何配置二、PHPstrom配置三、Xdebug的使用1.面板功能解释2.调试功能详解 四、Xdebug原理 前言 软件调试是泛指重现软件缺陷问题,定位和 查找问题根源,最终解决问题的过程,编写的程序不可能一直不出错&#xff0c;所以调试很重要调试通常有如…

西安阿里云代理商:阿里云服务器的可扩展性和弹性如何?是否支持按需付费?

西安阿里云代理商&#xff1a;阿里云服务器的可扩展性和弹性如何&#xff1f;是否支持按需付费&#xff1f;   一、阿里云服务器的可扩展性   阿里云作为业界知名的云服务提供商&#xff0c;其服务器具有极强的可扩展性。可扩展性主要体现在以下几方面&#xff1a;   1. …

小米note3刷机-从miui12刷回miui9

小米note3刷机-从miui12刷回miui9 文章目录 小米note3刷机-从miui12刷回miui9解除BL锁进入开发者模式遇到的问题解决BootLoader无法连接电脑的问题 导包 3月份原本想买一部小米6回来刷机,结果发现小米6的二手价格有一点点high。然后就选了一个 大平版 小米note3 但是直到昨天…

SpringBoot 如何使用 Logback 进行日志记录

SpringBoot 如何使用 Logback 进行日志记录 在开发 Web 应用程序时&#xff0c;日志记录是非常重要的一部分。日志可以帮助我们跟踪应用程序的运行情况&#xff0c;并帮助我们快速地排查问题。在 SpringBoot 中&#xff0c;我们可以使用 Logback 进行日志记录。Logback 是一款…

F407/103MAP文件

认识MAP文件 MDK编译工程&#xff0c;会生成一些中间文件&#xff08;如.o、.axf、.map 等&#xff09;&#xff0c;最终生成hex文件&#xff0c;以便下载到MCU上面执行。这些文件分为 11 个类型&#xff0c;其中4种文件比较重要。 比如&#xff1a; 本文主要讲解map文件。 map…

第四章 死锁

目录 一、死锁的概念 1.1 什么是死锁 1.2 死锁、饥饿、死循环的区别 1.2.1 死锁 1.2.2 饥饿 1.2.3 死循环 1.2.4 三者间的异同 1.3 死锁产生的必要条件 1.3.1 互斥条件 1.3.2 不剥夺条件 1.3.3 请求和保持条件 1.3.4 循环等待条件 1.4 什么时候会发生死锁 1.5 …