C语言实践——通讯录(3)(文件版)
首先感谢上一篇博客的大佬们的点赞,非常感谢!!!
目录
前言
一、需要添加的功能
1.增加保存数据函数——可以保存数据到文件中
主要逻辑:
注意事项:
代码实现:
2.修改初始化函数——新增载入数据模块
主要逻辑:
注意事项:
代码实现:
二、完整的代码实现
三、运行效果展示
前言
上一篇c语言实践博文我们将静态版的通讯录改成动态版,弥补了空间固定的尴尬。但是这个版本的通讯录还是不完整的。为什么呢?想想看,当我们运行程序,输入信息,退出程序,再进入程序,之前的数据还在不在了?不在了。因为那些数据是临时存储在内存中的,下次打开时已经被清理了。所以,我们需要在退出通讯录时,把数据存起来。在进入通讯录时,把数据加载进内存。
复习通讯录前两个版本请跳伞👇:
C语言实践——通讯录(2)(动态版)
c语言实践——通讯录(1)(静态版)
一、需要添加的功能
在程序已经开始执行的时候,通讯录的增删查改与之前逻辑一样,不用修改。但是在对通讯录进行操作之前,需要加载保存的数据的内容;在退出通讯录时,需要保存当前数据到文件中。
1.增加保存数据函数——可以保存数据到文件中
之前我们的代码执行逻辑为,当用户选择0时,先销毁通讯录释放内存,再退出。所以我们要在销毁通讯录之前加一个函数SaveContact,用来实现数据的保存。
主要逻辑:
- 打开文件。用fopen函数打开(或新建)一个文件,用文件指针接收。
- 写入数据。使用fwrite函数将Data中的数据保存到文件中。
- 关闭文件。用fclose函数关闭文件。
注意事项:
- 打开文件记得检验文件打开是否成功,若不成功,直接返回并打印错误信息。
- 写入数据时可以直接用Data数组名加整数的方式来表示要写入数据的指针。
- 关闭文件时最好将文件指针置为NULL。
代码实现:
//保存通讯录
void SaveContact(Contact* pc)
{//打开文件FILE* pf = fopen("contact.txt", "wb");if (pf == NULL){perror("SaveContact::fopen");return;}//数据写入int i = 0;for (i = 0; i < pc->size; i++){fwrite(pc->Data + i, sizeof(PeoInfo), 1, pf);}//关闭文件fclose(pf);pf = NULL;printf("保存成功\n");
}
2.修改初始化函数——新增载入数据模块
按照之前的代码逻辑,初始化时,要把大小,容量初始化为零,并为Data开辟一个空间。而现在,我们还需要将文件中的数据存载入Data指向的空间中。
主要逻辑:
- 初始化内容。和前面的版本一样,初始化变量,开辟内存。
- 打开文件。用fopen函数打开保存的文件,用文件指针接收。
- 读取数据。使用fread函数将文件中的数据读取存放到Data指向的空间。
- 关闭文件。用fclose函数关闭文件。
注意事项:
- 打开文件记得检验文件打开是否成功,若不成功,直接返回并打印错误信息。
- 读取数据时可以先用临时变量接收读取内容,接着检查空间容量,如果足够直接赋值,如果不够要开辟空间后再赋值。
- 每次读取数据不要忘记让数据大小size自增。
- 关闭文件时最好将文件指针置为NULL。
- 如果检查内存函数的定义放在后面,不要忘记事先声明。
代码实现:
//文件版本初始化
static void CheckCapacity(Contact* pc);
void LoadContact(Contact* pc)
{//打开文件FILE* pf = fopen("contact.txt", "rb");if (pf == NULL){perror("LoadContact::fopen");return;}//读文件PeoInfo tmp = { 0 };while (fread(&tmp, sizeof(PeoInfo), 1, pf)){CheckCapacity(pc);pc->Data[pc->size] = tmp;pc->size++;}//关闭文件fclose(pf);pf = NULL;printf("加载成功\n");
}
void InitContact(Contact* pc)
{pc->Data = (PeoInfo*)malloc(INIT_SIZE * sizeof(PeoInfo));if (pc->Data == NULL){printf("初始化通讯录失败:%s\n", strerror(errno));return;}pc->size = 0;pc->capacity = INIT_SIZE;//将文件中的数据加载到通讯录中LoadContact(pc);
}
二、完整的代码实现
通讯录最终的文件版本的代码在下面啦,有需自取。
contact.h
#include<stdlib.h>
#include<errno.h>#define MAX_NAME 15
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 5
#define ADD_SIZE 2
#define INIT_SIZE 4typedef struct PeoInfo
{char name[MAX_NAME];int age;char sex[MAX_SEX];char tele[MAX_TELE];char addr[MAX_ADDR];
}PeoInfo;typedef struct Contact
{PeoInfo* Data;int size;int capacity;
}Contact;void InitContact(Contact* pc);void ContactAdd(Contact* pc);void ShowContact(const Contact* pc);void ContactDel(Contact* pc);void ContactSearch(const Contact* pc);void ContactModify(Contact* pc);void ContactSort(Contact* pc);void DestroyContact(Contact* pc);void SaveContact(Contact* pc);void LoadContact(Contact* pc);
contact.c
#include "contact.h"static void CheckCapacity(Contact* pc);
void LoadContact(Contact* pc)
{//打开文件FILE* pf = fopen("contact.txt", "rb");if (pf == NULL){perror("LoadContact::fopen");return;}//读文件PeoInfo tmp = { 0 };while (fread(&tmp, sizeof(PeoInfo), 1, pf)){CheckCapacity(pc);pc->Data[pc->size] = tmp;pc->size++;}//关闭文件fclose(pf);pf = NULL;printf("加载成功\n");
}
void InitContact(Contact* pc)
{pc->Data = (PeoInfo*)malloc(INIT_SIZE * sizeof(PeoInfo));if (pc->Data == NULL){printf("初始化通讯录失败:%s\n", strerror(errno));return;}pc->size = 0;pc->capacity = INIT_SIZE;//将文件中的数据加载到通讯录中LoadContact(pc);
}void DestroyContact(Contact* pc)
{free(pc->Data);pc->Data = NULL;pc->size = 0;pc->capacity = 0;printf("释放内存\n");
}static void CheckCapacity(Contact* pc)
{if (pc->size == pc->capacity){PeoInfo* ptr = (PeoInfo*)realloc(pc->Data, (pc->capacity + 2) * sizeof(PeoInfo));if (ptr == NULL){printf("CheckCapacity:%s\n", strerror(errno));return;}pc->Data = ptr;pc->capacity += ADD_SIZE;printf("增容成功,当前容量:%d\n", pc->capacity);}
}void ContactAdd(Contact* pc)
{CheckCapacity(pc);printf("请输入名字>:");scanf("%s", pc->Data[pc->size].name);printf("请输入年龄>:");scanf("%d", &pc->Data[pc->size].age);printf("请输入性别>:");scanf("%s", pc->Data[pc->size].sex);printf("请输入电话>:");scanf("%s", pc->Data[pc->size].tele);printf("请输入地址>:");scanf("%s", pc->Data[pc->size].addr);pc->size++;printf("添加成功\n");
}static int FindByName(const Contact* pc, char name[MAX_NAME])
{int i = 0;for (i = 0; i < pc->size; i++){ //如果能找到if (0 == strcmp(pc->Data[i].name, name)){return i;}}//找不到return -1;
}void ShowContact(const Contact* pc)
{int i = 0;printf("%-15s %-5s %-5s %-11s %-5s", "姓名", "年龄", "性别", "电话", "地址\n");for (i = 0; i < pc->size; i++){printf("%-15s %-5d %-5s %-11s %-5s\n", pc->Data[i].name,pc->Data[i].age,pc->Data[i].sex,pc->Data[i].tele,pc->Data[i].addr);}}void ContactDel(Contact* pc)
{//1.找到要删除的数据下标char name[MAX_NAME];printf("请输入要删除的名字:>");scanf("%s", name);//如果找不到,提示后直接返回int pos = FindByName(pc,name);//按名字查找,找到了返回下标,找不到返回-1if (pos == -1){printf("找不到指定联系人\n");return;}//2.找到了就删除memmove(pc->Data + pos, pc->Data + pos + 1, (pc->size - 1 - pos)*sizeof(pc->Data[0]));pc->size--;printf("删除成功\n");
}void ContactSearch(const Contact * pc)
{char name[MAX_NAME];printf("请输入要查找的人的名字>:");scanf("%s", name);int pos = FindByName(pc, name);//按名字查找,找到了返回下标,找不到返回-1if (pos == -1){printf("找不到要查找的人\n");return;}printf("%-15s %-3s %-5s %-11s %-5s", "姓名", "年龄", "性别", "电话", "地址\n");printf("%-15s %-3d %-5s %-11s %-5s\n", pc->Data[pos].name,pc->Data[pos].age,pc->Data[pos].sex,pc->Data[pos].tele,pc->Data[pos].addr);
}void ContactModify(Contact* pc)
{//1.查找char name[MAX_NAME];printf("请输入要修改的人的名字>:");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("找不到要修改的联系人\n");return;}//修改printf("请输入名字>:");scanf("%s", pc->Data[pos].name);printf("请输入年龄>:");scanf("%d", &pc->Data[pos].age);printf("请输入性别>:");scanf("%s", pc->Data[pos].sex);printf("请输入电话>:");scanf("%s", pc->Data[pos].tele);printf("请输入地址>:");scanf("%s", pc->Data[pos].addr);printf("修改成功\n");
}int cmp_by_name(void* e1, void* e2)
{return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name);
}
void ContactSort(Contact* pc)
{qsort(pc->Data, pc->size, sizeof(PeoInfo), cmp_by_name);printf("排序成功\n");
}//保存通讯录
void SaveContact(Contact* pc)
{//打开文件FILE* pf = fopen("contact.txt", "wb");if (pf == NULL){perror("SaveContact::fopen");return;}//数据写入int i = 0;for (i = 0; i < pc->size; i++){fwrite(pc->Data + i, sizeof(PeoInfo), 1, pf);}//关闭文件fclose(pf);pf = NULL;printf("保存成功\n");
}
test.c
#include "contact.h"
void menu()
{printf("***************************\n");printf("***** 1.add *****\n");printf("***** 2.del *****\n");printf("***** 3.search *****\n");printf("***** 4.modify *****\n");printf("***** 5.show *****\n");printf("***** 6.sort *****\n");printf("***** 0.exit *****\n");printf("***************************\n");}
//使用枚举类型,提高可读性,比define更方便,
//define把字母替换成相应的数字,
//但是enum直接字母和数字完全相同
enum Option
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};int main()
{int input = 0;Contact con;//定义通讯录//初始化通讯录InitContact(&con);do{menu();printf("请选择>:");scanf("%d", &input);switch (input){case ADD:ContactAdd(&con);break;case DEL:ContactDel(&con);break;case SEARCH:ContactSearch(&con);break;case MODIFY:ContactModify(&con);break;case SHOW:ShowContact(&con);break;case SORT:ContactSort(&con);break;case EXIT:SaveContact(&con);DestroyContact(&con);printf("退出通讯录\n");break;default:printf("选择非法,请重新选择\n");break;}} while (input);return 0;
}
三、运行效果展示
首先创建文件(也可以不手动创建,因为fwrite函数会在最后创建出来,这里为了展示载入过程手动创建了文件。)
选择并新增信息。
容量不够自动扩容。
保存文件并退出。
再次进入通讯录并展示。
好啦,到目前位置,通讯录就基本完整了。该有的内容基本都有了。只不过实现的比较简单,还有一些没有细致考虑的东西,比如查找重名的问题无法解决。但是可以再新增函数,实现其他方式的查找,多项查找都满足才返回需要的数据的位置。这些不难,但是基本逻辑与FindByName函数类似,这里就不多讲了(没错,是我太懒想摸鱼)。
感谢大家的点赞关注,我会保持持续输出,动力满满!
相关文章:

C语言实践——通讯录(3)(文件版)
首先感谢上一篇博客的大佬们的点赞,非常感谢!!! 目录 前言 一、需要添加的功能 1.增加保存数据函数——可以保存数据到文件中 主要逻辑: 注意事项: 代码实现: 2.修改初始化函数——新…...

GPT撑腰,微软再战谷歌 | 大厂集体抢滩ChatGPT:谁真的有实力,谁在试点商业化?
国内互联网大厂已经很久没有这样的盛况了! 在各自领域成长为头部的互联网大厂们,近年来正在向“自留地”的纵深发展,正面交锋的机会并不多。直到大洋彼岸传来GPT的声音后,一下子抓住了大厂们的G点,他们仿佛听到了新一轮…...

【消息队列】细说Kafka消费者的分区分配和重平衡
消费方式 我们直到在性能设计中异步模式,一般要么是采用pull,要么采用push。而两种方式各有优缺点。 pull :说白了就是通过消费端进行主动拉去数据,会根据自身系统处理能力去获取消息,上有Broker系统无需关注消费端的…...

【Python从入门到人工智能】14个必会的Python内置函数(7)——打印输出(详细语法参考 + 参数说明 + 具体示例)| 附:Python输出表情包
你仔细想想,你和谁在一起的时候,最放得开、最自然、最舒服,又毫无顾忌,可以做回真实的你。那个人才是你心里最特别,最重要的人。 🎯作者主页: 追光者♂🔥 🌸个人简介: 💖[1] 计算机专业硕士研究生💖 🌟[2] 2022年度博客之星人工智能领域TOP4�…...

为什么要创建FAQ?这篇文章告诉你
什么是FAQ 通过上述的引入大家应该也了解到了,FAQ是为了“解决问题”而存在的。FAQ是英文Frequently Asked Questions的缩写,中文意思就是“经常问到的问题”,或者更通俗地叫做“常见问题解答”。FAQ是当前网络上提供在线帮助的主要手段&…...

基于html+css的盒子展示1
准备项目 项目开发工具 Visual Studio Code 1.44.2 版本: 1.44.2 提交: ff915844119ce9485abfe8aa9076ec76b5300ddd 日期: 2020-04-16T16:36:23.138Z Electron: 7.1.11 Chrome: 78.0.3904.130 Node.js: 12.8.1 V8: 7.8.279.23-electron.0 OS: Windows_NT x64 10.0.19044 项目…...

Python 无监督学习实用指南:1~5
原文:Hands-on unsupervised learning with Python 协议:CC BY-NC-SA 4.0 译者:飞龙 本文来自【ApacheCN 深度学习 译文集】,采用译后编辑(MTPE)流程来尽可能提升效率。 不要担心自己的形象,只关…...

2023 腾讯暑期实习申请经验分享
首先要向还在等我出 CMU 15-445 后面实验的同学们说声抱歉,这个系列可能暂时要停更啦。 一方面是博主最近课程和实验室方面的任务比较多,另一方面是有幸拿下了今年腾讯 WXG 后端开发的暑期实习 Offer,后面可能要提前学习一些工作中用到的框架…...

Protocol Buffers 介绍
Protocol Buffers Protocol Buffers ,协议缓冲区。什么是Protocol Buffers呢?或者我们简称PB 吧。那么Protocol Buffers 是一种与语言无关、与平台无关的可扩展机制,用于序列化结构化的数据。 example message Person {optional string nam…...

【模电实验】基尔霍夫定律、叠加定理和戴维南定理验证实验
实验目的 验证基尔霍夫电流定律(KCL)和电压定律(KVL)加深对该定理的理解验证叠加定理,加深对该定理的理解验证戴维南定理,掌握有源二端口网络的开路电压,短路电流和入端等效电阻的测定方法通过实…...

java某百货店POS积分管理系统_积分点更新生成以及通票回收处理
百货店是生活中不可缺少的一部分,为了给顾客提供更方便的服务平台以及更好的服务质量,而设计了POS积分管理系统。百货店通过点积分的管理获得顾客更好的信誉,增加客户流量,获得更多的利益。在百货店经营的过程中,每天的…...

Flutter 常用指令
1.flutter create app_01 :创建一个新的Flutter项目 2.flutter run:运行应用程序 3.flutter run -d <deviceId>:运行指定模拟器或者真机 4.flutter devices:查看计算机上的真机设备和IOS模拟器 5.flutter emulators&…...

定义全局变量property与getprop
authordaisy.skye的博客_CSDN博客-Qt,嵌入式,Linux领域博主 adb调试 adb shell getprop .adb logcat 报错 init: sys_prop: permission denied uid:1006 name:ro.camera.gc02m1 在linux驱动中查找 find ./ -name *.c | xargs grep -n "property_set" find ./ -n…...

双目三维测距(python)
文章目录 1. 双目检测1.1 调用相机1.2 分割画面 2. 双目标定2.1 相机标定2.2 获取参数 3. 双目测距3.1 立体校正3.1.1 校正目的3.1.2 校正方法3.1.2 相关代码 3.2 立体匹配和视差计算3.3 深度计算3.4 注意事项 4. 完整代码 代码打包下载: 链接1:https://…...

数据结构|二叉树的三种遍历方式,你掌握了几种?
目录 1、遍历方式 2、前序遍历 3、中序遍历 1、遍历方式 学习二叉树的结构,最简单的方式就是遍历二叉树。遍历二叉树就是通过某条线路对二叉树的各个结点进行一次访问,访问的方法有三种分为前序遍历、中序遍历、后续遍历,层序遍历它们的遍…...

Direct3D 12——灯光——法向量
a:平面法线着色 b:顶点法线着色 c:像素着色 平面法线(face normal,由于在计算机几何学中法线是有方向的向量,所以也有将normal译作法向量) 是 一种描述多边形朝向(即正交于多边形上所有点)的单位向量。 曲面法线&a…...

软考-信息系统工程(五)
信息系统工程 Garlan和Shaw对通用软件架构风格进行了分类,他们将软件架构分为:(曾经考过1分选择题 区分) 数据流风格:数据流风格包括批处理序列和管道/过滤器两种风格。调用/返回风格:调用/返回风格包括主程序/子程序、数据抽象和面向对象,以及层次结构…...

解决谷歌翻译不能使用的问题
今天登录国外网站,发现谷歌翻译已无法正常使用,网上最多的方法就是更改host文件,在host内增加ip地址,但是经常失效,经常手动更改增加ip着实烦恼,还有可能有别的错误。 最终解决方式是:登录GitH…...

Insomnia 简单使用方法
文章目录 1. 新建工程2. 新建若干文件夹3. 设置环境变量4. 授权以及进行请求的链式调用 (chaining requests)4. 1 解决办法 14. 2 解决办法 2 Insomnia 同 Postman, 用于测试后端 endpoint,很容易使用。 使用步骤如下: 1. 新建工程 2. 新建若…...

2023接口自动化测试,完整入门篇
1. 什么是接口测试 顾名思义,接口测试是对系统或组件之间的接口进行测试,主要是校验数据的交换,传递和控制管理过程,以及相互逻辑依赖关系。其中接口协议分为HTTP,WebService,Dubbo,Thrift,Socket等类型,测试类型又主…...

2023年股票代持行业研究报告
第一章 股票代持概述 1.1 基本概念 股票代持,或称委托持股,是指实际出资人与名义出资人达成以下约定:名义出资人作为名义股东,在股东名册等公司工商登记信息上出现,而实际上由实际出资人出资并享有投资权益。 股票代…...

《Netty》从零开始学netty源码(三十九)之PoolSubPage的内存分配
目录 PoolSubPage.allocategetNextAvail方法toHandle方法removeFromPool方法 PoolSubPage.allocate 上一篇我们介绍了PoolSubPage的简单知识,当我们需要PoolSubPage的内存时可调用allocate方法查找可分配二进制的位置,具体的源码过程如下: …...

【目标检测论文阅读笔记】Reducing Label Noise in Anchor-Free Object Detection
(Augmentation for small object detection) Abstract 当前的 anchor-free无锚目标检测器 将空间上落在真值框预定义中心区域内的所有特征标记为正。这种方法会在训练过程中产生 标签噪声,因为这些 正标记的特征中的一些 可能位于背景或遮挡…...

金融数字新型基础设施创新开放联合体今日成立
4月18日,“金融数字新型基础设施创新开放联合体”(以下简称:联合体)在上海成立。联合体由上海银行、复旦大学金融科技研究院、中电金信共同发起,首批成员单位汇聚产业链与供给侧的中坚力量:国泰君安证券、太…...

编程语言的发展史
编程语言处在不断的发展和变化中,从最初的机器语言发展到如今的2500种以上的高级语言,每种语言都有其特定的用途和不同的发展轨迹。编程语言并不像人类自然语言发展变化一样的缓慢而又持久,其发展是相当快速的,这主要是计算机硬件…...

巧用千寻位置GNSS软件|点测量采集技巧
点测量是测量中重要的节点,在测量工作的信息处理分析中发挥着重要作用。本期将给各位带来使用千寻位置GNSS软件采集地形点、控制点、快速点、连续点、房角点和倾斜点的操作技巧。 地形点 地形点的设置如图 5.1-9所 示,每次采集一个点,该点需要…...

DHCP原理与配置
目录 一、DHCP工作原理 1)了解DHCP服务 使用DHCP的好处 DHCP的分配方式 2)DHCP的租约过程 分为四个步骤 二、DHCP服务器的配置 1)检查并且安装dhcp有关软件包 2)查看系统的配置文件,并且利用好官方给的参考案…...

软件测试今天你被内卷了吗?
认识一个人,大专学历非计算机专业的,是前几年环境好的时候入的行,那时候软件测试的要求真的很低,他那时好像是报了个班,然后入门的,但学的都是些基础,当时的他想的也简单,反正也能拿…...

做完自动化测试,但别让不会汇报毁了你...
pytest 是一个成熟的全功能Python测试工具,可以帮助您编写更好的程序。它与 python 自带的 unittest 测试框架类似,但 pytest 使用起来更简洁和高效,并且兼容 unittest 框架。pytest 能够支持简单的单元测试和复杂的功能测试,pyte…...

企业级信息系统开发讲课笔记2.4 利用MyBatis实现条件查询
文章目录 零、本节学习目标一、查询需求二、打开MyBatisDemo项目三、对学生表实现条件查询(一)创建学生映射器配置文件(二)配置学生映射器文件(三)创建学生映射器接口(四)测试学生映…...