【C】—文件版本通讯录的实现

2023-02-20 10:51:37 浏览数 (1)

关于C语言的知识放在专栏:C 小菜坤日常上传gitee代码:https://gitee.com/qi-dunyan ❤❤❤ 个人简介:双一流非科班的一名小白,期待与各位大佬一起努力!

目录

  • 思路
  • 代码实现
  • 完整代码(可自取)

思路

在前面的文章中,已经讲解了动态版本的通讯录的实现,但是动态通讯录存在一个致命缺陷,就是它不能自动保存数据,而前面一篇文章中学到了数据持久化的方法之一:即把数据存放在磁盘文件上,便可以实现数据持久化。 具体应该如何做呢? 假如我们在退出的时候,通过文件操作,把我们所写的数据存在磁盘文件里,然后我们再进行下一次的使用的时候,在初始化阶段就从磁盘中读取这些数据,这不就实现了。

代码实现

代码的实现并不困难,只不过是在动态内存版本的基础上进行了一些文件操作,用来保存和读取数据。 从文件中读取信息(初始化阶段完成)

代码语言:javascript复制
//读通讯录文件信息
//size_t fread(void* ptr, size_t size, size_t count, FILE* stream)
void Load_Contact(struct contact* p)
{
	//二进制读
	FILE* pfR = fopen("通讯录.txt", "rb");
	if (pfR == NULL)
	{
		//读取失败打印错误报告
		perror("Load_Contact::fopen");
		return;
	}
	struct message pf = { 0 };
	//fread返回值为读取的完整的元素个数,这里读取成功返回1,失败0
	while (fread(&pf, sizeof(pf), 1, pfR))
	{
		//判断是否增容
		check_capacity(p);
		p->data[p->sz] = pf;
		p->sz  ;
	}
	//关闭文件
	fclose(pfR);
	pfR = NULL;
}

//初始化通讯录
void Init_contact(struct contact* p)
{
	assert(p);
	//开辟空间
	p->data =(struct message*) malloc(DEFAULT_SZ * sizeof(struct message));
	//假如开辟失败,报错
	if (p->data == NULL)
	{
		printf("%sn", strerror(errno));
		return;
	}
	p->sz = 0;
	p->capacity = DEFAULT_SZ;
	//加载通讯录信息
	Load_Contact(p);
}

这里在动态版本的基础上,在初始化阶段加入了一个Load_Contact()函数,这个是用来以二进制读的方式打开文件,并且把读取到的信息放在结构体pf里,然后再将pf赋值到p指向的data数组的下标为size空间。

将数据写入文件(退出时保存信息)

这一步是为了将我们本次所写的数据,写入到文件中去,以备下一次打开时好从中读取数据。具体代码如下:

代码语言:javascript复制
//写数据(保存通讯录信息)
//size_t fwrite(const void* ptr, size_t size, size_t count, FILE* stream)
void Storage_Contact(struct contact* p)
{
	//二进制形式写
	FILE* pfW = fopen("通讯录.txt", "wb");
	if (pfW == NULL)
	{
		//失败则打印错误报告
		perror("Storage_Contact::fopen");
		return;
	}
	int i = 0;
	//将sz个数据都写入到文件中
	for (i = 0; i < p->sz; i  )
	{
		fwrite(p->data   i, sizeof(struct message), 1, pfW);
	}
	//关闭
	fclose(pfW);
	pfW = NULL;
}

正是有了这一步,我们在退出时会创建一个文件来保存信息。供下一次读取。如以下视频所示:在我第二次运行程序时,上一次的数据已经加载完毕了。实现了数据持久化。

文件版本通讯录(退出可保存信息)

完整代码(可自取)

.h头文件

代码语言:javascript复制
#pragma once
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
#include<Windows.h>
#include<errno.h>

#define NAME 20
#define SEX 5
#define TELE 12
#define ADDR 30

#define DEFAULT_SZ 3//初始容量
#define INC_SZ 2//扩容

//联系人信息
struct message
{
	//姓名
	char name[NAME];
	//性别
	char sex[SEX];
	//电话
	char tele[TELE];
	//住址
	char addr[ADDR];
	//年龄
	int age;
};
//通讯录
struct contact
{
	struct message* data;
	int sz;//个数
	int capacity;//通讯录容量
};
//初始化通讯录
void Init_contact(struct contact* p);
//动态增加联系人
void Add_contact(struct contact* p);
//显示联系人
void Show_contact(const struct contact* p);
//删除联系人
void Dele_contact(struct contact* p);
//修改联系人信息
void revise_contact(struct contact* p);
//查找联系人信息
void Find_contact(const struct contact* p);
//排序联系人
void Sort_contact(struct contact* p);
//清空联系人
void Clean(struct contact* p);
//释放空间
void Destory_contact(struct contact* p);
//保存数据
void Storage_Contact(struct contact* p);

.c源文件(函数定义)

代码语言:javascript复制
#include"Contact_exe.h"

//是否判断增容
int check_capacity(struct contact* p)
{
	//当联系人个数 == 通讯录容量时,增容INC_SZ个内存空间
	if (p->sz == p->capacity)
	{
		struct message* ptr = (struct message*)realloc(p->data, (p->capacity   INC_SZ) * sizeof(struct message));
		if (ptr == NULL)//判断是否增容失败
		{
			printf("%sn", strerror(errno));
			return 0;
		}
		else
		{
			p->data = ptr;//增容成功,data就指向这块新开辟的空间
			p->capacity  = INC_SZ;//容量 =INC_SZ
			//printf("增容成功!n");
			return 1;
		}
	}
	//不需要增容
	else
		return 1;
}

//读通讯录文件信息
//size_t fread(void* ptr, size_t size, size_t count, FILE* stream)
void Load_Contact(struct contact* p)
{
	//二进制读
	FILE* pfR = fopen("通讯录.txt", "rb");
	if (pfR == NULL)
	{
		//读取失败打印错误报告
		perror("Load_Contact::fopen");
		return;
	}
	struct message pf = { 0 };
	//fread返回值为读取的完整的元素个数,这里读取成功返回1,失败0
	while (fread(&pf, sizeof(pf), 1, pfR))
	{
		//判断是否增容
		check_capacity(p);
		p->data[p->sz] = pf;
		p->sz  ;
	}
	//关闭文件
	fclose(pfR);
	pfR = NULL;
}

//初始化通讯录
void Init_contact(struct contact* p)
{
	assert(p);
	//开辟空间
	p->data =(struct message*) malloc(DEFAULT_SZ * sizeof(struct message));
	//假如开辟失败,报错
	if (p->data == NULL)
	{
		printf("%sn", strerror(errno));
		return;
	}
	p->sz = 0;
	p->capacity = DEFAULT_SZ;
	//加载通讯录信息
	Load_Contact(p);
}


//增加联系人
void Add_contact(struct contact* p)
{
	assert(p);
	if (0 == check_capacity(p))
	{
		printf("%sn", strerror(errno));
		return;
	}
	printf("请输入姓名:->");
	scanf("%s", p->data[p->sz].name);

	printf("请输入性别:->");
	scanf("%s", p->data[p->sz].sex);

	printf("请输入电话:->");
	scanf("%s", p->data[p->sz].tele);

	printf("请输入住址:->");
	scanf("%s", p->data[p->sz].addr);

	printf("请输入年龄:->");
	scanf("%d", &(p->data[p->sz].age));
	system("cls");

	printf("增加成功!n");
	printf("n");
	p->sz  ;
}
//显示联系人
void Show_contact(const struct contact* p)
{
	assert(p);
	int i = 0;
	printf("%-20st%-5st%-12st%-20st%-5sn", "姓名", "性别", "电话", "住址", "年龄");
	for (i = 0; i < p->sz; i  )
	{
		printf("%-20st%-5st%-12st%-20st%-5dn", p->data[i].name,
			p->data[i].sex,
			p->data[i].tele,
			p->data[i].addr,
			p->data[i].age);
	}
}
int find_name(const struct contact* p, char arr[])
{
	assert(p);
	int i = 0;
	for (i = 0; i < p->sz; i  )
	{
		if (0 == strcmp(p->data[i].name, arr))
			return i;
	}
	return -1;
}
//删除联系人
void Dele_contact(struct contact* p)
{
	assert(p);
	char del_name[NAME];
	printf("请输入要删除联系人的姓名:->");
	scanf("%s", del_name);
	//查找该联系人
	int ret = find_name(p, del_name);
	if (ret == -1)
		printf("查无此人!n");
	else
	{
		int j = 0;
		for (j = ret; j < p->sz - 1; j  )
		{
			p->data[j] = p->data[j   1];
		}
		p->sz--;
		system("cls");

		printf("删除成功!n");
		printf("n");
	}
}
//修改菜单栏
void menu_()
{
	printf("***********************************n");
	printf("******   1、修改联系人姓名   ******n");
	printf("******   2、修改联系人电话   ******n");
	printf("******   3、修改联系人年龄   ******n");
	printf("******   4、修改联系人住址   ******n");
	printf("******   5、修改联系人性别   ******n");
	printf("******   0、返回主菜单       ******n");

}
//修改联系人信息
void revise_contact(struct contact* p)
{
	assert(p);
	char del_name[NAME];
	printf("请输入要修改信息的联系人的姓名:->");
	scanf("%s", del_name);
	int ret = find_name(p, del_name);
	if (ret == -1)
	{
		printf("查无此人!n");
		printf("n");
	}
	else
	{
		int in_put = 0;
		do
		{
			menu_();
			scanf("%d", &in_put);
			switch (in_put)
			{
			case 1:
				printf("请输入修改后的姓名:->");
				scanf("%s", p->data[ret].name);
				system("cls");

				printf("姓名修改成功!n");
				break;
			case 2:
				printf("请输入修改后的电话:->");
				scanf("%s", p->data[ret].tele);
				system("cls");

				printf("电话修改成功!n");
				break;
			case 3:
				printf("请输入修改后的年龄:->");
				scanf("%d", &(p->data[ret].age));
				system("cls");

				printf("年龄修改成功!n");
				break;
			case 4:
				printf("请输入修改后的住址:->");
				scanf("%s", p->data[ret].addr);
				system("cls");

				printf("住址修改成功!n");
				break;
			case 5:
				printf("请输入修改后的性别:->");
				scanf("%s", p->data[ret].sex);
				system("cls");

				printf("性别修改成功!n");
				break;
			case 0:
				printf("取消修改!n");
				break;
			default:
				printf("输入错误!n");
				break;
			}
		} while (in_put);
	}
}
//查找联系人信息
void Find_contact(const struct contact* p)
{
	assert(p);
	char del_name[NAME];
	printf("请输入要查找联系人的姓名:->");
	scanf("%s", del_name);
	system("cls");

	//查找该联系人
	int ret = find_name(p, del_name);
	if (ret == -1)
		printf("查无此人!n");
	else
	{
		printf("%-20st%-5st%-12st%-20st%-5sn", "姓名", "性别", "电话", "住址", "年龄");
		printf("%-20st%-5st%-12st%-20st%-5dn", p->data[ret].name,
			p->data[ret].sex,
			p->data[ret].tele,
			p->data[ret].addr,
			p->data[ret].age);
	}
}
//排序菜单
void menu_sort()
{
	printf("******   1、姓名   ******n");
	printf("******   2、住址   ******n");
	printf("******   3、年龄   ******n");
	printf("******   4、性别   ******n");
	printf("******   0、退出   ******n");
}

//姓名排序
int cmp_name(const void* e1, const void* e2)
{
	return strcmp(((struct message*)e1)->name, ((struct message*)e2)->name);
}
//住址排序
int cmp_addr(const void* e1, const void* e2)
{
	return strcmp(((struct message*)e1)->addr, ((struct message*)e2)->addr);
}
//年龄排序
int cmp_age(const void* e1, const void* e2)
{
	return ((struct message*)e1)->age - ((struct message*)e2)->age;
}
//性别排序
int cmp_sex(const void* e1, const void* e2)
{
	return strcmp(((struct message*)e1)->sex, ((struct message*)e2)->sex);
}
//排序联系人
void Sort_contact(struct contact* p)
{
	int s = 0;
	do
	{
		//排序菜单
		menu_sort();
		printf("请选择排序类型:->");
		scanf("%d", &s);
		system("cls");
		switch (s)
		{
		case 1:
			qsort(p->data, p->sz, sizeof(struct message), cmp_name);
			printf("排序成功!n");
			break;
		case 2:
			qsort(p->data, p->sz, sizeof(struct message), cmp_addr);
			printf("排序成功!n");
			break;
		case 3:
			qsort(p->data, p->sz, sizeof(struct message), cmp_age);
			printf("排序成功!n");
			break;
		case 4:
			qsort(p->data, p->sz, sizeof(struct message), cmp_sex);
			printf("排序成功!n");
			break;
		case 0:
			printf("退出排序n");
			break;
		default:
			printf("输入有误!n");
			break;
		}

	} while (s);
}

//清空联系人
void Clean(struct contact* p)
{
	p->sz = 0;
	printf("清空成功!n");
}

//释放空间
void Destory_contact(struct contact* p)
{
	free(p->data);
	p->data=NULL;
	p->sz = 0;
	p->capacity = 0;
}
//写数据(保存通讯录信息)
//size_t fwrite(const void* ptr, size_t size, size_t count, FILE* stream)
void Storage_Contact(struct contact* p)
{
	//二进制形式写
	FILE* pfW = fopen("通讯录.txt", "wb");
	if (pfW == NULL)
	{
		//失败则打印错误报告
		perror("Storage_Contact::fopen");
		return;
	}
	int i = 0;
	for (i = 0; i < p->sz; i  )
	{
		fwrite(p->data   i, sizeof(struct message), 1, pfW);
	}
	fclose(pfW);
	pfW = NULL;
}

.c源文件(测试)

代码语言:javascript复制
#include"Contact_exe.h"

void menu()
{
	printf("--------------------------------------------------------------n");
	printf("---------   1、增加联系人       2、删除指定联系人    ---------n");
	printf("---------   3、修改联系人信息   4、查找联系人        ---------n");
	printf("---------   5、排序联系人       6、显示已有联系人    ---------n");
	printf("---------   0、退出并保存信息   7、清空联系人        ---------n");
	printf("--------------------------------------------------------------n");
}

int main()
{
	int input = 0;
	//创建通讯录
	struct contact con;
	//初始化通讯录(读之前通讯录的信息)
	Init_contact(&con);
	do
	{
		menu();
		printf("请选择:->");
		scanf("%d", &input);
		system("cls");
		switch (input)
		{
		case 1:
			//增加联系人
			Add_contact(&con);
			break;
		case 2:
			//删除联系人
			Show_contact(&con);
			Dele_contact(&con);
			break;
		case 3:
			//修改联系人信息
			Show_contact(&con);
			revise_contact(&con);
			break;
		case 4:
			//查找联系人信息
			Find_contact(&con);
			break;
		case 5:
			//排序联系人信息
			Show_contact(&con);
			Sort_contact(&con);
			break;
		case 6:
			//显示联系人
			system("cls");
			Show_contact(&con);
			printf("n");
			break;
		case 7:
			//清空联系人
			Clean(&con);
			break;
		case 0:
			//保存数据
			Storage_Contact(&con);
			//释放空间
			Destory_contact(&con);
			break;
		default:
			printf("输入错误!n");
			break;
		}
	} while (input);
	return 0;
}

0 人点赞