目录
前言
文件存储版的通讯录 是在通讯录的基础上添加了文件操作等
在了解通讯录(文件存储版)之前,我们得好好学习一下 文件操作
只有掌握了操作文件,才能明白文件存储版的通讯录
一.基本思路
1.通讯录是由多人的信息组合,信息:姓名,年龄,性别,地址等。
2.通讯录的大小,存放的人数,此使用动态存储,更加方便存储和利用空间。
3.通讯录的基本功能:增加联系人、删除联系人、查找联系人,修改联系人
4.保存通讯录文档,方便下次使用。
注:通讯录分三种:静态、动态、文件三种。而此文章讲述的是动态存储的文件版通讯录。
静态:大小固定,存储的人数有明确限制,无法改变,使用数组实现。
动态:存储人数可以调节,可以随着人数的增加而增加,选择一个初始大小,之后可进行扩充操作, 可更好的利用空间。
文件:该通讯录是在以上两种通讯录之一上加上存储文件的操作,在程序执行结束后都无法保存,录入的信息在程序结束时就会消失。为了保存录入的信息,可以通过文件操作来实现。
二.代码的实现
2.1通讯录菜单
菜单能够实现和用户的交互。需要选择增、删、查、改的功能。
所以,通讯录需要一个菜单。
代码如下:
void menu()
{
printf("*********************************\n");
printf("*******1.添加 2.删除*******\n");
printf("*******3.查找 4.修改*******\n");
printf("*******5.显示 6.保存*******\n");
printf("*******0.退出 *******\n");
printf("*********************************\n");
}
//使用函数指针数组
void (*(p[7]))(Contact*) = { Exit,AddContact,DeleteContact,SearchContact,ModifyContact,PrintContact,SaveContact };
int main()
{
//Contact cl; 这是下面代码结构体的
//InitContact(&cl);
int input;
do {
menu();
printf("请输入->");
scanf("%d", &input);
if (input <= 6)
p[input](&cl);
else
printf("输入错误\n");
}while (input != 0);
}
为了方便,我们使用了函数指针数组,void (*(p[7]))(Contact*)
2.2通讯录的定义及功能
通讯录需要姓名、性别、年龄及电话,此时我们需要使用结构体来定义。还需要定义函数:增、删、查、改、打印学生信息、保存通讯录。
代码如下:
#include<stdio.h>
#include<malloc.h>
#include<stdlib.h>
#include<string.h>
typedef struct PeoInfo
{
char name[10];//姓名
int age; //年龄
char sex[3]; //性别
int phone[12];//电话
}PeoInfo;
typedef struct Contact
{
int size; //当前存储的人数
int capacity; //通讯录容量大小
contact* data; //结构体指针,访问个人信息
}Contact;
//初始化通讯录
void InitContact(Contact* pc);
//增加联系人
void AddContact(Contact* pc);
//删除联系人
void DeleteContact(Contact* pc);
//查找联系人
void SearchContact(Contact* pc);
//修改联系人
void ModifyContact(Contact* pc);
//显示通讯录
void PrintContact(Contact* pc);
//扩容通讯录
void CheckCapacity(Contact* pc);
//保存信息到文件
void SaveContact(Contact* pc);
//文件信息传递到通讯录
void LoadContact(Contact* pc);
//销毁空间,退出程序
void Exit(Contact* pc);
2.3函数实现
注:当通讯录退出的时候,把信息写到文件。
当通讯录初始化后,加载文件的信息到通讯录中。
2.3.1初始化通讯录
通讯录是动态的,所以需要扩容,给一定空间。
void InitContact(Contact* pc)
{
// 此时通讯录是空的,应先为通讯录分配空间
pc->data = (PeoInfo*)malloc(sizeof(PeoInfo) * 4);
// 如果分配成功,将通讯录的size设为0,capacity设为初识大小
pc->size = 0;
pc->capacity = 4;
//将文档的信息传递到动态内存里,函数实现在下方
LoadContact(pc);
}
2.3.2文件信息传递到通讯录里
将文件的信息加载到通讯录里
void LoadContact(Contact* pc)
{
// 以读的形式打开文件
FILE* pf = fopen("contact.txt", "rb");
if (pf == NULL)
{
perror("fopen");
return;
}
// 将文件中的内容加载到通讯录中
// 这里用fread函数,当fread函数读取的联系人信息数为0时,说明读取结束
PeoInfo tmp = { 0 }; // 定义一个联系人信息的结构体,便于读取
int i = 0;
while (fread(&tmp, sizeof(PeoInfo), 1, pf))
{
CheckCapacity(pc); // 在这个函数中查看数组是否需要扩容,若需要,则扩容
pc->data[i] = tmp;
pc->size++;
i++;
}
// 关闭文件
fclose(pf);
pf = NULL;
return;
}
2.3.3扩容通讯录
添加联系人之前,需要判断通讯录是否满员,满员则需要扩容
void CheckCapacity(Contact* pc)
{
// 判断通讯录是否已满,若满,进行扩容
if (pc->size == pc->capacity)
{
PeoInfo* tmp = (PeoInfo*)realloc(pc, sizeof(PeoInfo) * 4);
if (tmp == NULL)
{
printf("realloc fail\n");
return;
}
pc->data = tmp;
// 若扩容成功,增大capacity
pc->capacity *= 2;
}
}
2.3.4增加联系人
需要添加联系人的各项信息。
void AddContact(Contact* pc)
{
int num = 0;
printf("添加人数:");
scanf("%d", &num);
// 输入要添加的联系人的信息
// 这里pc->data为结构体数组,pc->data[pc->size]为其中的元素,也就是某一个联系人的信息
for (int i = 0; i < num; i++)
{
//判断通讯录是否满人
CheckCapacity(pc);
printf("请输入名字\n");
scanf("%s", pc->data[pc->size].name);
printf("请输入性别\n");
scanf("%s", pc->data[pc->size].sex);
printf("请输入年龄\n");
scanf("%d", &pc->data[pc->size].age);
printf("请输入电话\n");
scanf("%s", pc->data[pc->size].phone);
pc->size++; // 将存入的联系人的数量加1
}
}
2.3.5删除联系人
需要找到该联系人的下标值,然后进行删除。
void DeleteContact(Contact* pc)
{
int ret = 0;//记录寻找的下标值
printf("请输入要删除的联系人的名字\n");
char name[20];
scanf("%s", name);
// 定义一个新函数find,用来查找是否有这个联系人
// 如果有,返回联系人的下标,如果没有,返回-1
for (int i = 0; i < pc->size; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
ret = i; break;
}
}
if (ret)
{
printf("不存在");
}
else {
for (int i = ret; i < pc->size - 1; i++)
{
pc->data[i] = pc->data[i + 1];
}
pc->size--;
}
}
2.3.6查询联系人
找到下标值,从而得到联系人的信息
void SearchContact(Contact* pc)
{
int ret = 0;
printf("请输入要查找的联系人的名字\n");
char name[20];
scanf("%s", name);
// 利用已经定义的find函数进行查找
for (int i = 0; i < pc->size; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
ret = i; break;
}
}
if (ret)
{
printf("没有找到该联系人\n");
}
else
{
// 如果找到,打印该联系人的信息,首先打印五个标题
printf("%-10s\t%-10s\t%-5s\t%-15s\n", "姓名", "性别", "年龄", "电话");
printf("%-10s\t%-10s\t%-5d\t%-15s%\n",
pc->data[ret].name,
pc->data[ret].sex,
pc->data[ret].age,
pc->data[ret].phone
);
}
}
2.3.7修改联系人
找到下标值,从而进行各项数据的修改
void ModifyContact(Contact* pc)
{
int ret = 0;
printf("请输入要查找的联系人的名字\n");
char name[20];
scanf("%s", name);
for (int i = 0; i < pc->size; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
ret = i; break;
}
}
if (ret)
{
printf("没有找到该联系人\n");
}
else
{
printf("请输入名字\n");
scanf("%s", pc->data[ret].name);
printf("请输入性别\n");
scanf("%s", pc->data[ret].sex);
printf("请输入年龄\n");
scanf("%d", &pc->data[ret].age);
printf("请输入电话\n");
scanf("%s", pc->data[ret].phone);
}
}
2.3.8打印通讯录
打印每个人的具体数据
// 在这个函数内打印所有联系人的信息
void PrintContact(Contact* pc)
{
// 首先打印五个标题
printf("%-10s\t%-10s\t%-5s\t%-15s\n", "姓名", "性别", "年龄", "电话");
// 然后用for循环打印所有联系人的信息
for (int i = 0; i < pc->size; i++)
{
printf("%-10s\t%-10s\t%-5d\t%-15s\n",
pc->data[i].name,
pc->data[i].sex,
pc->data[i].age,
pc->data[i].phone
);
}
}
2.3.9信息保留在文件中
程序退出时,要将销毁空间,并且把数据存入文件里,方便下次调用
void SaveContact(Contact* pc)
{
FILE* p = fopen("contact.txt", "wb");
if (p == NULL)
{
perror("SaveContact");
}
else {
int i = 0;
for (i; i < pc->size; i++)
{
fwrite(pc->data + i, sizeof(PeoInfo), 1, p);
}
fclose(p);
p = NULL;
printf("保存成功");
}
}
2.3.10销毁空间退出程序
void Exit(Contact* pc)
{
//销毁空间
free(pc->data);
pc->data = NULL;
pc->size = 0;
pc->capacity = 0;
}
三.完整代码
和上面的代码比起来,连贯了Contact.c的各个函数实现。
3.1Text.c文件
#include"Contact.h"
void menu()
{
printf("*********************************\n");
printf("*******1.添加 2.删除*******\n");
printf("*******3.查找 4.修改*******\n");
printf("*******5.显示 6.保存*******\n");
printf("*******0.退出 *******\n");
printf("*********************************\n");
}
//使用函数指针数组
void (*(p[7]))(Contact*) = { Exit,AddContact,DeleteContact,SearchContact,ModifyContact,PrintContact,SaveContact };
int main()
{
//定义结构体
Contact cl;
InitContact(&cl);
int input;
do {
menu();
printf("请输入->");
scanf("%d", &input);
if (input <= 6)
p[input](&cl);
else
printf("输入错误\n");
} while (input != 0);
}
3.2Contact.h文件
#include<stdio.h>
#include<malloc.h>
#include<stdlib.h>
#include<string.h>
typedef struct PeoInfo
{
char name[10];//姓名
int age; //年龄
char sex[3]; //性别
int phone[12];//电话
}PeoInfo;
typedef struct Contact
{
int size; //当前存储的人数
int capacity; //通讯录容量大小
PeoInfo* data; //结构体指针,访问个人信息
}Contact;
//初始化通讯录
void InitContact(Contact* pc);
//增加联系人
void AddContact(Contact* pc);
//删除联系人
void DeleteContact(Contact* pc);
//查找联系人
void SearchContact(Contact* pc);
//修改联系人
void ModifyContact(Contact* pc);
//显示通讯录
void PrintContact(Contact* pc);
//扩容通讯录
void CheckCapacity(Contact* pc);
//销毁并且保存通讯录
void SaveContact(Contact* pc);
void LoadContact(Contact* pc);
void Exit(Contact* pc);
3.3Contact.c文件
#include"Contact.h"
void InitContact(Contact* pc)
{
// 此时通讯录是空的,应先为通讯录分配空间
pc->data = (PeoInfo*)malloc(sizeof(PeoInfo) * 4);
// 如果分配成功,将通讯录的size设为0,capacity设为初识大小
pc->size = 0;
pc->capacity = 4;
//将文档的信息传递到动态内存里,函数实现在下方
LoadContact(pc);
}
void LoadContact(Contact* pc)
{
// 以读的形式打开文件
FILE* pf = fopen("contact.txt", "rb");
if (pf == NULL)
{
perror("fopen");
return;
}
// 将文件中的内容加载到通讯录中
// 这里用fread函数,当fread函数读取的联系人信息数为0时,说明读取结束
PeoInfo tmp = { 0 }; // 定义一个联系人信息的结构体,便于读取
int i = 0;
while (fread(&tmp, sizeof(PeoInfo), 1, pf))
{
CheckCapacity(pc); // 在这个函数中查看数组是否需要扩容,若需要,则扩容
pc->data[i] = tmp;
pc->size++;
i++;
}
// 关闭文件
fclose(pf);
pf = NULL;
return;
}
void CheckCapacity(Contact* pc)
{
// 判断通讯录是否已满,若满,进行扩容
if (pc->size == pc->capacity)
{
PeoInfo* tmp = (PeoInfo*)realloc(pc, sizeof(PeoInfo) * 4);
if (tmp == NULL)
{
printf("realloc fail\n");
return;
}
pc->data = tmp;
// 若扩容成功,增大capacity
pc->capacity *= 2;
}
}
void AddContact(Contact* pc)
{
int num = 0;
printf("添加人数:");
scanf("%d", &num);
// 输入要添加的联系人的信息
// 这里pc->data为结构体数组,pc->data[pc->size]为其中的元素,也就是某一个联系人的信息
for (int i = 0; i < num; i++)
{
//判断通讯录是否满人
CheckCapacity(pc);
printf("请输入名字\n");
scanf("%s", pc->data[pc->size].name);
printf("请输入性别\n");
scanf("%s", pc->data[pc->size].sex);
printf("请输入年龄\n");
scanf("%d", &pc->data[pc->size].age);
printf("请输入电话\n");
scanf("%s", pc->data[pc->size].phone);
pc->size++; // 将存入的联系人的数量加1
}
}
void DeleteContact(Contact* pc)
{
int ret = 0;//记录寻找的下标值
printf("请输入要删除的联系人的名字\n");
char name[20];
scanf("%s", name);
// 定义一个新函数find,用来查找是否有这个联系人
// 如果有,返回联系人的下标,如果没有,返回-1
for (int i = 0; i < pc->size; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
ret = i; break;
}
}
if (ret)
{
printf("不存在");
}
else {
for (int i = ret; i < pc->size - 1; i++)
{
pc->data[i] = pc->data[i + 1];
}
pc->size--;
}
}
void SearchContact(Contact* pc)
{
int ret = 0;
printf("请输入要查找的联系人的名字\n");
char name[20];
scanf("%s", name);
// 利用已经定义的find函数进行查找
for (int i = 0; i < pc->size; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
ret = i; break;
}
}
if (ret)
{
printf("没有找到该联系人\n");
}
else
{
// 如果找到,打印该联系人的信息,首先打印五个标题
printf("%-10s\t%-10s\t%-5s\t%-15s\n", "姓名", "性别", "年龄", "电话");
printf("%-10s\t%-10s\t%-5d\t%-15s%\n",
pc->data[ret].name,
pc->data[ret].sex,
pc->data[ret].age,
pc->data[ret].phone
);
}
}
void ModifyContact(Contact* pc)
{
int ret = 0;
printf("请输入要查找的联系人的名字\n");
char name[20];
scanf("%s", name);
// 利用已经定义的find函数进行查找
for (int i = 0; i < pc->size; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
ret = i; break;
}
}
if (ret)
{
printf("没有找到该联系人\n");
}
else
{
printf("请输入名字\n");
scanf("%s", pc->data[ret].name);
printf("请输入性别\n");
scanf("%s", pc->data[ret].sex);
printf("请输入年龄\n");
scanf("%d", &pc->data[ret].age);
printf("请输入电话\n");
scanf("%s", pc->data[ret].phone);
}
}
// 在这个函数内打印所有联系人的信息
void PrintContact(Contact* pc)
{
// 首先打印五个标题
printf("%-10s\t%-10s\t%-5s\t%-15s\n", "姓名", "性别", "年龄", "电话");
// 然后用for循环打印所有联系人的信息
for (int i = 0; i < pc->size; i++)
{
printf("%-10s\t%-10s\t%-5d\t%-15s\n",
pc->data[i].name,
pc->data[i].sex,
pc->data[i].age,
pc->data[i].phone
);
}
}
void SaveContact(Contact* pc)
{
FILE* p = fopen("contact.txt", "wb");
if (p == NULL)
{
perror("SaveContact");
}
else {
int i = 0;
for (i; i < pc->size; i++)
{
fwrite(pc->data + i, sizeof(PeoInfo), 1, p);
}
fclose(p);
p = NULL;
printf("保存成功");
}
}
void Exit(Contact* pc)
{
//销毁空间
free(pc->data);
pc->data = NULL;
pc->size = 0;
pc->capacity = 0;
}
总结
只有不断的掌握知识,我们才能敲代码得心应手,在不断的学习里,逐渐进步。
-------------小菜TQ02