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等类型,测试类型又主…...

srs linux
下载编译运行 git clone https:///ossrs/srs.git ./configure --h265on make 编译完成后即可启动SRS # 启动 ./objs/srs -c conf/srs.conf # 查看日志 tail -n 30 -f ./objs/srs.log 开放端口 默认RTMP接收推流端口是1935,SRS管理页面端口是8080,可…...

【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...

CVE-2020-17519源码分析与漏洞复现(Flink 任意文件读取)
漏洞概览 漏洞名称:Apache Flink REST API 任意文件读取漏洞CVE编号:CVE-2020-17519CVSS评分:7.5影响版本:Apache Flink 1.11.0、1.11.1、1.11.2修复版本:≥ 1.11.3 或 ≥ 1.12.0漏洞类型:路径遍历&#x…...

无人机侦测与反制技术的进展与应用
国家电网无人机侦测与反制技术的进展与应用 引言 随着无人机(无人驾驶飞行器,UAV)技术的快速发展,其在商业、娱乐和军事领域的广泛应用带来了新的安全挑战。特别是对于关键基础设施如电力系统,无人机的“黑飞”&…...

FFmpeg avformat_open_input函数分析
函数内部的总体流程如下: avformat_open_input 精简后的代码如下: int avformat_open_input(AVFormatContext **ps, const char *filename,ff_const59 AVInputFormat *fmt, AVDictionary **options) {AVFormatContext *s *ps;int i, ret 0;AVDictio…...

恶补电源:1.电桥
一、元器件的选择 搜索并选择电桥,再multisim中选择FWB,就有各种型号的电桥: 电桥是用来干嘛的呢? 它是一个由四个二极管搭成的“桥梁”形状的电路,用来把交流电(AC)变成直流电(DC)。…...

uni-app学习笔记三十五--扩展组件的安装和使用
由于内置组件不能满足日常开发需要,uniapp官方也提供了众多的扩展组件供我们使用。由于不是内置组件,需要安装才能使用。 一、安装扩展插件 安装方法: 1.访问uniapp官方文档组件部分:组件使用的入门教程 | uni-app官网 点击左侧…...
python打卡day49@浙大疏锦行
知识点回顾: 通道注意力模块复习空间注意力模块CBAM的定义 作业:尝试对今天的模型检查参数数目,并用tensorboard查看训练过程 一、通道注意力模块复习 & CBAM实现 import torch import torch.nn as nnclass CBAM(nn.Module):def __init__…...

CVE-2023-25194源码分析与漏洞复现(Kafka JNDI注入)
漏洞概述 漏洞名称:Apache Kafka Connect JNDI注入导致的远程代码执行漏洞 CVE编号:CVE-2023-25194 CVSS评分:8.8 影响版本:Apache Kafka 2.3.0 - 3.3.2 修复版本:≥ 3.4.0 漏洞类型:反序列化导致的远程代…...
Cursor AI 账号纯净度维护与高效注册指南
Cursor AI 账号纯净度维护与高效注册指南:解决限制问题的实战方案 风车无限免费邮箱系统网页端使用说明|快速获取邮箱|cursor|windsurf|augment 问题背景 在成功解决 Cursor 环境配置问题后,许多开发者仍面临账号纯净度不足导致的限制问题。无论使用 16…...