两个文件分别保存若干学生数据(包括学号、姓名和成绩),且按成绩升序排列。编写程序,把两个文件归并到第三个文件,使第三个文件中的学生数据也按成绩升序排列。

一、题目

程序功能:两个非空二进制文件 a1.dat 和 a2.dat,分别保存若干学生数据(包括学号、姓名和成绩),且按成绩升序排列。编写程序,把 a1.dat 和 a2.dat 两个文件归并到 a3.dat 文件,使得 a3.dat 文件中的学生数据也按成绩升序排列。要求如下:

① 定义 mergeFile()函数:将两个二进制文件归并成一个按成绩升序排列的新文件。

② 定义 main()函数:先打开二进制文件 a1.dat、a2.dat、a3.dat,再调用 mergeFile()函数将 a1.dat、a2.dat 两个文件归并到按成绩升序排列的 a3.dat 文件,最后输出 a3.dat 文件内容。

二、程序清单

#include<stdio.h>
#include<stdlib.h>

typedef struct student {
	char id[30];//学号
	char name[20];//姓名
	char score[10];//成绩
	struct student* next;
};

student* ReadFile(FILE* pfile)//读取文件信息并存入链表中并返回该链表
{
	student* L = (student*)malloc(sizeof(student));
	student* r = L;
	while (!feof(pfile))
	{
		student* p = (student*)malloc(sizeof(student));
		fscanf(pfile, "%s %s %s\n", &p->id, p->name, p->score);
		r->next = p;
		r = r->next;
	}
	r->next = NULL;
	return L;
}

student* mergeFile(FILE* p1, FILE* p2, FILE* p3)//将两个二进制文件归并成一个按成绩升序排列的新文件
{
	student* L1 = ReadFile(p1);//调用函数
	student* L2 = ReadFile(p2);//调用函数
	student* r1 = L1->next;
	student* r2 = L2->next;
	student* L3 = (student*)malloc(sizeof(student));
	student* r3 = L3;
	while (r1 && r2)//若两链表均未结尾,则进入循环
	{
		if (atoi(r1->score) >= atoi(r2->score))//将成绩较低的学生信息链入新链表的结尾
		{
			fprintf(p3, "%s %s %s\n", r2->id, r2->name,r2->score);
			r3->next = r2;
			r3 = r3->next;
			r2 = r2->next;
		}
		else
		{
			fprintf(p3, "%s %s %s\n", r1->id, r1->name, r1->score);
			r3->next = r1;
			r3 = r3->next;
			r1 = r1->next;
		}
	}
	if (r1)//将较长的链表中未接入的部分接入新链表的末尾
	{
		while (r1)
		{
			atoi(r1->score);
			fprintf(p3, "%s %s %s\n", r1->id, r1->name, r1->score);
			r3->next = r1;
			r3 = r3->next;
			r1 = r1->next;
		}
	}
	else
		while (r2)
		{
			atoi(r2->score);
			fprintf(p3, "%s %s %s\n", r2->id, r2->name, r2->score);
			r3->next = r2;
			r3 = r3->next;
			r2 = r2->next;
		}
	return L3;//返回升序排列后的链表头指针
}

void PrintList(student* L)//输出单链表内容
{
	student* p = L->next;
	printf("学号\t姓名\t成绩\n");
	while (p)
	{
		printf("%s\t%s\t%s\n", p->id, p->name, p->score);
		p = p->next;
	}
}

int main()
{
	FILE* p1;
	FILE* p2;
	FILE* p3;
	if ((p1 = fopen("a1.dat", "rb")) == NULL)//读模式
	{
		printf("文件a1.dat打开失败!");
		exit(0);
	}
	if ((p2 = fopen("a2.dat", "rb")) == NULL)//读模式
	{
		printf("文件a2.dat打开失败!");
		exit(0);
	}
	if ((p3 = fopen("a3.dat", "wb")) == NULL)//写模式
	{
		printf("文件a3.dat打开失败!");
		exit(0);
	}
	PrintList(mergeFile(p1, p2, p3));//调用函数
	if (fclose(p1))
	{
		printf("不能正常关闭文件a1.dat!");
		exit(0);
	}
	if (fclose(p2))
	{
		printf("不能正常关闭文件a2.dat!");
		exit(0);
	}
	if (fclose(p3))
	{
		printf("不能正常关闭文件a3.dat!");
		exit(0);
	}
	return 0;
}

三、运行结果

 

  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值