通讯录-文件操作版

news/2025/2/27 17:13:03/

之前我们写过通讯录-动态开辟版,但里面的数据录入后,若退出程序,里面的数据也就跟着一起销毁,无法保存,所以今天我们来写可建议将通讯录信息保存起来的版本,这只要在原来的基础上加以改进就可以了。

首先,我们先将菜单,枚举常量和switch语句中加入储存选项

//test.c
void menu()
{printf("*****************************************\n");printf("********** 1.add          2.del    ******\n");printf("********** 3.search       4.modify ******\n");printf("********** 5.show         6.sort   ******\n");printf("********** 7.save         0.exit   ******\n");printf("*****************************************\n");
}
//contact.h
enum option
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT,SAVE
};
//test.c
do{menu();//打印菜单printf("请选择:>");scanf("%d", &input);//选择功能switch (input){case ADD://添加信息AddContact(&con);break;case DEL://删除信息DelContact(&con);break;case SEARCH://查找信息SearchContact(&con);break;case MODIFY://修改信息ModifyContact(&con);break;case SHOW://显示信息ShowContact(&con);break;case SORT://整理信息SortContact(&con);break;case SAVE:SaveContact(&con);//加入储存通讯录信息的函数//保存信息break;case EXIT:SaveContact(&con);//在退出程序前也进行信息存储DestroyContact(&con);printf("退出通讯录\n");break;default:printf("选择错误\n");break;}} while (input);

保存信息函数的实现

void SaveContact(Contact* ps)
{FILE* pfwrite = fopen("contact.dat", "wb");if (pfwrite == NULL){printf("%s\n", strerror(errno));return;}int i = 0;for (i = 0; i < ps->size;i++){fwrite(&(ps->date[i]), sizeof(PeoInfo), 1, pfwrite);}printf("保存成功\n");fclose(pfwrite);pfwrite = NULL;
}

程序关闭后,当程序再一次执行,我想之前保存的信息任然存在。

在创建通讯录后初始化

所以初始化函数InitaContact开辟空间后要把文件中已经存放通讯录中的信息加载到通讯录中

读取信息函数的实现

void LoadContact(Contact* ps)
{PeoInfo tmp;FILE* pfread = fopen("contact.dat", "rb");if (pfread == NULL){printf("LoadContact:%s\n", strerror(errno));}//读取文件存放到通讯录中fread(&tmp,sizeof(PeoInfo),1,pfread);fcolse(pfread);pfread = NULL;
}

写到这会出现两个问题

  1. fread读取数据读到什么时候停下来?

  1. 读到这个信息最终要放到这个通讯录里面去,如果要放的通讯录信息大于通讯录开辟的空间呢?

问题1:

fread读取数据读到什么时候停下来?

fread函数的返回值是fread真实读到的元素个数

如果count的值是10,fread的返回值是5,说明fread没有那么多元素可以读了

如果fread的返回值比count要小,说明fread读完了

让count=1;

如果读完了,fread的返回值就是0了。

void LoadContact(Contact* ps)
{PeoInfo tmp;FILE* pfread = fopen("contact.dat", "rb");if (pfread == NULL){printf("LoadContact:%s\n", strerror(errno));}//读取文件存放到通讯录中while (fread(&tmp, sizeof(PeoInfo), 1, pfread)){}fcolse(pfread);pfread = NULL;
}

问题2:

读到这个信息最终要放到这个通讯录里面去,如果要放的通讯录信息大于通讯录开辟的空间呢?

信息放到通讯录里就要保证通讯录的空间是有效的,足够的,这就需要用到CheckCapacity函数

void LoadContact(Contact* ps)
{PeoInfo tmp;FILE* pfread = fopen("contact.dat", "rb");if (pfread == NULL){printf("LoadContact:%s\n", strerror(errno));}//读取文件存放到通讯录中while (fread(&tmp, sizeof(PeoInfo), 1, pfread)){//读到了就把元素放到通讯录里//放到通讯录就要保证这个空间是有效的,这就需要用到CheckCapacity函数CheckCapacity(ps);ps->date[ps->size] = tmp;ps->size++;}fcolse(pfread);pfread = NULL;
}

完整代码

test.c

#define _CRT_SECURE_NO_WARNINGS
#include"contact.h"void menu()
{printf("*****************************************\n");printf("********** 1.add          2.del    ******\n");printf("********** 3.search       4.modify ******\n");printf("********** 5.show         6.sort   ******\n");printf("********** 7.save         0.exit   ******\n");printf("*****************************************\n");
}int main()
{int input;//创建通讯录construct Contact con;//con就是通讯录,里面包含:date指针和size,capacity//初始化通讯录InitContact(&con);do{menu();//打印菜单printf("请选择:>");scanf("%d", &input);//选择功能switch (input){case ADD://添加信息AddContact(&con);break;case DEL://删除信息DelContact(&con);break;case SEARCH://查找信息SearchContact(&con);break;case MODIFY://修改信息ModifyContact(&con);break;case SHOW://显示信息ShowContact(&con);break;case SORT://整理信息SortContact(&con);break;case SAVE:SaveContact(&con);//保存信息break;case EXIT:SaveContact(&con);DestroyContact(&con);printf("退出通讯录\n");break;default:printf("选择错误\n");break;}} while (input);return 0;
}

contact.c

#define _CRT_SECURE_NO_WARNINGS
#include"contact.h"void InitContact(struct Contact* ps)
{ps->date = (PeoInfo*)malloc(defuault_sz * sizeof(PeoInfo));if (ps->date == NULL){return 0;}ps->size = 0;ps->capacity = defuault_sz;//把文件中已经存放的通讯率信息加载到通讯录中LoadContact(ps);}void CheckCapacity(Contact* ps);void LoadContact(Contact* ps)
{PeoInfo tmp;FILE* pfread = fopen("contact.dat", "rb");if (pfread == NULL){printf("LoadContact:%s\n", strerror(errno));}//读取文件存放到通讯录中while (fread(&tmp, sizeof(PeoInfo), 1, pfread)){//读到了就把元素放到通讯录里//放到通讯录就要保证这个空间是有效的,这就需要用到CheckCapacity函数CheckCapacity(ps);ps->date[ps->size] = tmp;ps->size++;}fclose(pfread);pfread = NULL;
}void CheckCapacity(Contact* ps)
{if (ps->size == ps->capacity){PeoInfo* ptr = (PeoInfo*)realloc(ps->date, (ps->capacity + 2) * sizeof(PeoInfo));if (ptr != NULL){ps->date = ptr;ps->capacity += 2;printf("增容成功\n");}else{printf("增容失败\n");}}
}void AddContact(struct Contact* ps)
{//检测当前通讯录的容量//1.如果满了,就增加空间//2.如果不满,就啥事不干CheckCapacity(ps);//增加数据printf("请输入名字:>");scanf("%s", ps->date[ps->size].name);printf("请输入病历号:>");scanf("%s", ps->date[ps->size].id);printf("请输入症状:>");scanf("%s", ps->date[ps->size].symptom);ps->size++;printf("添加成功\n");
}void ShowContact(const struct Contact* ps)
{if (ps->size == 0){printf("通讯录为空\n");}else{int i = 0;printf("%-20s\t%-20s\t%-30s\n", "姓名", "病历号", "症状");for (i = 0; i < ps->size; i++){printf("%-20s", ps->date[i].name);printf("\t%-20s", ps->date[i].id);printf("\t%-30s\n", ps->date[i].symptom);}}
}static int FindByName(const struct Contact* ps, char name[max_name])
{int i = 0;for (i = 0; i < ps->size; i++){if (0 == strcmp(ps->date[i].name, name)){return i;}}return -1;
}void DelContact(struct Contact* ps)
{char name[max_name];printf("请输入要删除人的名字:>");scanf("%s", name);//1.查找要删除的人在什么位置//找到返回下标//找不到返回-1;int pos= FindByName(ps, name);//2.删除if (pos==-1){printf("查无此人\n");}else{//删除数据int j = 0;for(j=pos;j<ps->size-1;j++){ ps->date[j] = ps->date[j + 1];}ps->size--;printf("删除成功\n");}}void SearchContact(const struct Contact* ps)
{char name[max_name];printf("请输入要查找人的名字:>");scanf("%s", name);int pos = FindByName(ps, name);if (pos == -1){printf("查无此人\n");}else{printf("%-20s\t%-20s\t%-30s\n", "姓名", "病历号", "症状");printf("%-20s", ps->date[pos].name);printf("\t%-20s", ps->date[pos].id);printf("\t%-30s\n", ps->date[pos].symptom);}
}void ModifyContact(struct Contact* ps)
{char name[max_name];printf("请输入要修改人的名字:>");scanf("%s", name);int pos=FindByName(ps, name);if (pos == -1){printf("要修改人的信息不存在\n");}else{printf("请输入名字:>");scanf("%s", ps->date[pos].name);printf("请输入病历号:>");scanf("%s", ps->date[pos].id);printf("请输入症状:>");scanf("%s", ps->date[pos].symptom);printf("修改完成\n");}
}static int cmp_contact_by_id(const void* e1, const void* e2)
{return strcmp(((struct Contact*)e1)->date->id , ((struct Contact*)e2)->date->id);}void SortContact(struct Contact* ps)
{int sz = ps->size;qsort(ps->date, sz, sizeof(ps->date[0]), cmp_contact_by_id);printf("排序成功\n");
}void DestroyContact(Contact* ps)
{free(ps->date);ps->date = NULL;
}void SaveContact(Contact* ps)
{FILE* pfwrite = fopen("contact.dat", "wb");if (pfwrite == NULL){printf("SaveContact:%s\n", strerror(errno));return;}int i = 0;for (i = 0; i < ps->size;i++){fwrite(&(ps->date[i]), sizeof(PeoInfo), 1, pfwrite);}printf("保存成功\n");fclose(pfwrite);pfwrite = NULL;
}

contact.h

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>
#include<stdlib.h>//#define MAX 1000#define defuault_sz 3
#define max_name 20
#define max_id 24
#define max_symptom 100enum option
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT,SAVE
};
typedef struct PeoInfo
{char name[max_name];char id[max_id];char symptom[max_symptom];
}PeoInfo;typedef struct Contact
{struct PeoInfo *date;//存放个人信息int size;//记录当前已有的元素个数;int capacity;//当前通讯录的最大容量
}Contact;
//函数声明
void InitContact(struct Contact* ps);void AddContact(struct Contact* ps);void ShowContact(const struct Contact* ps);void DelContact(struct Contact* ps);void SearchContact(const struct Contact* ps);void ModifyContact(struct Contact*ps);void SortContact(struct Contact* ps);void DestroyContact(Contact* ps);void SaveContact(Contact* ps);//加载文件中的信息到通讯录
void LoadContact(Contact*ps);

以上就是本篇文章的内容了,很感谢你能看到这里

如果觉得内容对你有帮助的话,不妨点个关注

我会继续更新更高质量的内容,我们一同学习,一同进步!


http://www.ppmy.cn/news/33376.html

相关文章

经典PID控制算法原理以及优化思路

文章目录0、概念1、理解2、实现3、优化4、引用0、概念 PID算法是工业应用中最广泛算法之一&#xff0c;在闭环系统的控制中&#xff0c;可自动对控制系统进行准确且迅速的校正。PID控制&#xff0c;即Proportional – Integral(I) – Derivative(D) Control, 实际上是三种反馈…

GCC 基本使用

GCC 原名为GNU C语言编译器&#xff08;GNU C Compiler&#xff09;GCC(GNU Compiler Collection, GNU 编译器套件) 是由GNU开发的编程语言编译器。GNU 编译器套件包括C、C、Objective-C、Java、Ada和Go语言前端&#xff0c;也包括了这些语言的库&#xff08;如libstdc、libgcj…

数据结构--二叉树

目录1.树概念及结构1.1数的概念1.2数的表示2.二叉树概念及结构2.1二叉树的概念2.2数据结构中的二叉树2.3特殊的二叉树2.4二叉树的存储结构2.4.1顺序存储2.4.2链式存储2.5二叉树的性质3.堆的概念及结构3.1堆的实现3.1.1堆的创建3.1.2堆的插入3.1.3堆顶的删除3.1.4堆的代码实现3.…

【C语言】位段(详解)

目录1. 什么是位段2. 位段的内存分配1. 什么是位段 结构体中可以位为单位来指定其成员所占内存长度&#xff0c;这种以位为单位的成员称为“位段”或“位域”( bit field) 。利用位段能够用较少的位数存储数据。 位段的好处&#xff1a; 位段可以使数据单元节省储存空间位段可…

Spring Profiles and @Profile

1. Overview In this tutorial, we’ll focus on introducing Profiles in Spring. Profiles are a core feature of the framework — allowing us to map our beans to different profiles — for example, dev, test, and prod. We can then activate different profiles…

Keil5安装和使用小记

随着keil版本的更新&#xff0c;一些使用问题一随之产生。本文针对安装目前最新版本keil软件和使用问题做一些总结。 目录1 Keil5下载&安装1.1 官网下载链接1.2 软件安装1.2.1 安装说明1.2.2 关于 51 和 ARM 共存的问题1.3 软件破解2 pack包安装 & 破解2.1 下载2.2 安装…

Go语言必知必会100问题-02 减少代码的嵌套层数

减少代码的嵌套层数 软件开发中的“心智模型”用于描述开发人员在编码时心理活动&#xff0c;每段代码其实是人在编写这段代码时的心智模型投射&#xff0c;不能把代码看成是客观的存在&#xff0c;而是主观的产物&#xff0c;参合了当时心理活动或各种直觉感知。在编程时&…

特征建模之FiBiNet

FiBiNet: Combine Feature importance and Bilinear feature Interaction for Click-Through Rate Prediction https://arxiv.org/abs/1905.0943 一、特征建模的重要性 推荐领域的深度CTR模型中的参数主要由两部分构成&#xff1a;特征Embedding参数和MLP层参数&#xff0c;假…