【C语言】通讯录3.0 (文件存储版)

前言

通讯录是一种记录联系人信息的工具,包括姓名、电话号码、电子邮件地址、住址等。
文章的一二三章均于上一篇相同,可以直接看第四章改造内容。
此通讯录是基于通讯录2.0(动态增长版)的基础上进行增加文件操作功能,请先看系列文章第二篇,再看本篇博客。
****** 有需要源代码,见文章末尾 ******


系列文章目录

第一篇:【C语言】通讯录1.0 (静态版)
第二篇:【C语言】通讯录2.0 (动态增长版)
第三篇:【C语言】通讯录3.0 (文件存储版)


文章目录

  • 前言
  • 系列文章目录
  • 一、什么是通讯录
  • 二、静态版、动态增长版和文件存储版的区别
    • 1. 静态版
    • 2. 动态增长版
    • 3. 文件存储版
  • 三、通讯录模块组成(图文)
    • 1. 通讯录文件构成
    • 2. 通讯录个人信息
    • 3. 通讯录功能模块
  • 四、改造通讯录2.0(改造目标)
    • 1. 文件存储通讯录要求
    • 2. 需要添加的功能模块
  • 五、如何改进(代码演示)
    • 1. 建立文件
    • 2. 写入文件
    • 3. 读取文件
  • 六、所有文件代码
    • 1.头文件
    • 2. 函数文件
    • 3. 逻辑测试文件
  • 总结


一、什么是通讯录

通讯录是一种记录联系人信息的工具,包括姓名、电话号码、电子邮件地址、住址等。通讯录可以帮助人们管理自己的联系人,让人们更轻松地与他人保持联系。通讯录可以在手机、电脑、笔记本等设备上保存,也可以在云端储存和同步,方便用户随时查看和更新联系人信息。

二、静态版、动态增长版和文件存储版的区别

C语言静态版、动态增长版和文件存储版的区别如下:

1. 静态版

  1. 静态版:在程序编译时就确定了内存大小,程序运行期间内存大小不会发生变化,因此对于需要处理大量数据或者不确定数据大小的情况不适用。

2. 动态增长版

  1. 动态增长版:可以在程序运行期间根据需要动态增加内存大小,因此适用于处理不确定数据大小的情况。但是动态增长的内存需要手动释放,否则会导致内存泄漏。

3. 文件存储版

  1. 文件存储版:将数据存储在文件中,可以持久保存数据并随时读取。但是存储在文件中的数据需要进行IO操作,因此相比于内存操作来说效率较低。此外,文件存储版不适用于需要频繁修改的数据。

三、通讯录模块组成(图文)

1. 通讯录文件构成

在这里插入图片描述

2. 通讯录个人信息

在这里插入图片描述

3. 通讯录功能模块

在这里插入图片描述

四、改造通讯录2.0(改造目标)

1. 文件存储通讯录要求

  1. 将通讯录的联系人信息可以保存在文件里
  2. 保存到通讯录里的文件以二进制文件保存
  3. 第二次打开时,读取文件里上一次保存的联系人信息

2. 需要添加的功能模块

在这里插入图片描述

五、如何改进(代码演示)

1. 建立文件

  • 使用文件操作 fopen 函数来建立一个文件
FILE* pf = fopen("data.dat","wb");
	if (pf == NULL)
	{
		perror("SaveContact");
		return;

	}

2. 写入文件

  • 使用文件操作 fwrite 函数
  • 使用for循环,依次将通讯录内容写入文件
//写入文件
void SaveContact(Contact* pc)
{
	FILE* pf = fopen("data.dat","wb");
	if (pf == NULL)
	{
		perror("SaveContact");
		return;

	}
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		fwrite(pc->data+i, sizeof(PeoInfo), 1, pf);
	}
	fclose(pf);
	pf = NULL;
	printf("保存成功\n");
}

3. 读取文件

  • 使用文件操作 fread 函数
  • 使用whlie循环,先判断fread的返回值,为真进入循环
  • 使用扩容函数 determine( ) 来判断内存是否充足,进行增容
//读取文件
void ReadContact(Contact* pc)
{
	FILE* pf = fopen("data.dat", "rb");
	if (pf == NULL)
	{
		perror("ReadContact");
		return;

	}
	PeoInfo tem = { 0 };
	while (fread (&tem, sizeof(PeoInfo), 1, pf))
	{
		if (determine(pc) == 0)
		{
			return;
		}
		pc->data[pc->sz] = tem;
		pc->sz++;
	}

	fclose(pf);
	pf = NULL;
}

六、所有文件代码

1.头文件

#pragma once
#include<stdio.h>
#include<assert.h>
#include<string.h>
#include<stdlib.h>

#define MAX 100
#define NAME 10
#define SEX  5
#define TELE 12
#define ADDR 30
//使用枚举  定义选择   
enum OPTION
{
	EXIT,//0
	ADD,
	DEL,
	SEARCH,
	MODIFY,
	SHOW,
	SORT
};


//个人信息类型声明
typedef struct PeoInfo
{
	char name[NAME];
	int age;
	char sex[SEX];
	char tele[TELE];
	char addr[ADDR];
}PeoInfo;

//建立通讯录
//静态版
//typedef struct Contact
//{
//	PeoInfo data[MAX]; //通讯录数量
//	int sz;				//目前通讯录内的人数	
//}Contact;
//动态版
typedef struct Contact
{
	PeoInfo* data;
	int sz;
	int capacity;

}Contact;

//函数声明

//初始化通讯录
void InitContact(Contact* pc);

//增加联系人
void AddContact(Contact* pc);

//显示所有联系人的信息
void ShowContact(const Contact* pc);

//删除指定联系人
void DelContact(Contact* pc);

//查找指定联系人
void SearchContact(const Contact* pc);


//修改指定联系人
void ModifyContact(Contact* pc);


//释放内存

void DestroyContact(Contact* pc);

//保存通讯录
void SaveContact(Contact* pc);

2. 函数文件

#define _CRT_SECURE_NO_WARNINGS 1
#include "addbook.h"
void ReadContact(Contact* pc);
//静态版
//void InitContact(Contact* pc)
//{
//	assert(pc);   //断言
//	memset(pc->data, 0, sizeof(pc->data));   //内存函数  data初始化为0  
//	pc->sz = 0;
//}
int determine(Contact* pc)
{
	assert(pc);
	if (pc->capacity == pc->sz)
	{
		PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + 2) * sizeof(PeoInfo));

		if (ptr == NULL)
		{
			perror("determine");

			return 0;
		}
		else
		{
			pc->capacity += 2;
			pc->data = ptr;
			printf("增容成功\n");
			return 1;
		}


	}
	return 1;
}
//动态版
void InitContact(Contact* pc)
{
	assert(pc);
	pc->data = (PeoInfo*)malloc(3 * sizeof(PeoInfo));
	if (pc->data == NULL)
	{
		perror("InitContact");
		return;
	}
	pc->capacity = 3;
	pc->sz = 0;
	ReadContact(pc);
}
//动态版
void AddContact(Contact* pc)
{
	assert(pc);       //断言
	if (determine(pc) == 0)
	{
		return;
	}
	printf("请输入姓名:>");
	scanf("%s", pc->data[pc->sz].name);
	printf("请输入年龄:>");
	scanf("%d", &(pc->data[pc->sz].age));
	printf("请输入性别:>");
	scanf("%s", pc->data[pc->sz].sex);
	printf("请输入电话:>");
	scanf("%s", pc->data[pc->sz].tele);
	printf("请输入地址:>");
	scanf("%s", pc->data[pc->sz].addr);
	pc->sz++;    //通讯录加1
	printf("联系人增加成功\n");
}




//静态版
//void AddContact(Contact* pc)
//{
//	assert(pc);       //断言
//	if (pc->sz == MAX)         //如果通讯录已经满了  则返回
//	{
//		printf("通讯录已满,无法添加\n");
//		return;
//	}
//	printf("请输入姓名:>");
//	scanf("%s", pc->data[pc->sz].name);
//	printf("请输入年龄:>");
//	scanf("%d", &(pc->data[pc->sz].age));
//	printf("请输入性别:>");
//	scanf("%s", pc->data[pc->sz].sex);
//	printf("请输入电话:>");
//	scanf("%s", pc->data[pc->sz].tele);
//	printf("请输入地址:>");
//	scanf("%s", pc->data[pc->sz].addr);
//	pc->sz++;    //通讯录加1
//	printf("联系人增加成功\n");
//}
//搜索名字找通讯录函数
static int Findname(const Contact* pc, char na[])
{
	int i = 0;
	assert(pc && na);
	for (i = 0; i < pc->sz; i++)
	{
		if (strcmp(pc->data[i].name, na) == 0)

		{
			return i;
		}

	}
	return -1;
}
void DelContact(Contact* pc)
{
	if (pc->sz == 0)
	{
		printf("通讯录为空\n");

		return;
	}
	char name[NAME] = { 0 };
	assert(pc);
	//输入要查找的联系人名字
	printf("请输入要查找的名字:>");
	scanf("%s", &name);
	//找到要查找的联系人
	int del = Findname(pc, name);
	//删除坐标位子的联系人 ,将后面的联系人进行代替其位置
	if (del == -1)
	{
		printf("找不到,此人不存在\n");
		return;
	}
	else
	{
		int i = 0;
		for (i = del; i < pc->sz; i++)
		{
			pc->data[i] = pc->data[i + 1];
		}
		pc->sz--;
	}

	printf("成功删除联系人\n");
}
void ShowContact(const Contact* pc)
{
	assert(pc);
	printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
			pc->data[i].name,
			pc->data[i].age,
			pc->data[i].sex,
			pc->data[i].tele,
			pc->data[i].addr);
	}
	printf("通讯录展示完毕\n");
}
void SearchContact(const Contact* pc)
{
	if (pc->sz == 0)
	{
		printf("通讯录为空\n");

		return;
	}
	char name[NAME] = { 0 };
	assert(pc);
	//输入要查找的联系人名字
	printf("请输入要查找的名字:>");
	scanf("%s", &name);
	//找到要查找的联系人
	int i = Findname(pc, name);
	if (i == -1)
	{
		printf("找不到,此人不存在\n");
		return;
	}
	else
	{
		printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
		printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
			pc->data[i].name,
			pc->data[i].age,
			pc->data[i].sex,
			pc->data[i].tele,
			pc->data[i].addr);
	}

	printf("成功找到联系人\n");
}
void ModifyContact(Contact* pc)
{
	assert(pc);
	char name[NAME] = { 0 };
	printf("请输入要修改人的名字:>");
	scanf("%s", &name);
	int mod = Findname(pc, name);
	if (mod == -1)
	{
		printf("找不到,不存在\n");
		return;
	}
	else
	{
		printf("请输入姓名:>");
		scanf("%s", pc->data[mod].name);
		printf("请输入年龄:>");
		scanf("%d", &(pc->data[mod].age));
		printf("请输入性别:>");
		scanf("%s", pc->data[mod].sex);
		printf("请输入电话:>");
		scanf("%s", pc->data[mod].tele);
		printf("请输入地址:>");
		scanf("%s", pc->data[mod].addr);

		printf("联系人修改成功\n");
	}
}
void DestroyContact(Contact* pc)
{
	free(pc->data);
	pc->data = NULL;
	pc->capacity = 0;
	pc->sz;
}

//写入文件
void SaveContact(Contact* pc)
{
	FILE* pf = fopen("data.dat","wb");
	if (pf == NULL)
	{
		perror("SaveContact");
		return;

	}
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		fwrite(pc->data+i, sizeof(PeoInfo), 1, pf);
	}
	fclose(pf);
	pf = NULL;
	printf("保存成功\n");
}
//读取文件
void ReadContact(Contact* pc)
{
	FILE* pf = fopen("data.dat", "rb");
	if (pf == NULL)
	{
		perror("ReadContact");
		return;

	}
	PeoInfo tem = { 0 };
	while (fread (&tem, sizeof(PeoInfo), 1, pf))
	{
		if (determine(pc) == 0)
		{
			return;
		}
		pc->data[pc->sz] = tem;
		pc->sz++;
	}

	fclose(pf);
	pf = NULL;
}

3. 逻辑测试文件

#define _CRT_SECURE_NO_WARNINGS 1
#include"addbook.h"
void menu()
{
	printf("********************************\n");
	printf("***** 1. ADD     2. DEL    *****\n");
	printf("***** 3. SEARCH  4. MODIFY *****\n");
	printf("***** 5. SHOW    6. SORT   *****\n");
	printf("***** 0. EXIT              *****\n");
	printf("********************************\n");

}

void test()
{
	int input = 0;
	Contact con;
	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:
			printf("功能待开发\n");
			break;
		case EXIT:
			SaveContact(&con);
			DestroyContact(&con);
			printf("成功退出通讯录\n");
			break;
		default:
			printf("输入错误,请重新输入\n");
			break;
		}
	} while (input);

}
int main()
{
	test();
	return 0;
}

总结

本期博客,是通讯录3.0(文件存储版),是对前面所学知识进行复习,编写通讯录时有助于理解自定义类型、动态内存管理及文件操作的学习和了解,通讯录的文章到这里就结束了,谢谢支持!!!


如这篇博客对大家有帮助的话,希望 三连 支持一下 !!! 如果有错误感谢大佬的斧正 如有 其他见解发到评论区,一起学习 一起进步。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/61341.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【多线程学习6】synchronized关键字

【多线程学习6】synchronized关键字 一、synchronized关键字是什么&#xff1f;有什么作用&#xff1f; synchronized关键字是Java线程同步的关键字&#xff0c;其可以修饰方法或代码块&#xff0c;并可以保证其修饰的方法或代码块在任意时刻只能有一个线程执行。 synchroni…

K8s集群安全机制

1.访问K8s集群的时候&#xff0c;需要经过三个步骤完成具体操作 &#xff08;1&#xff09;认证&#xff08;2&#xff09;鉴权&#xff08;授权&#xff09;&#xff08;3&#xff09;准入控制 进行访问的时候&#xff0c;过程中都要经过apiserver&#xff0c;apiserver做统…

java,python,c++有什么区别,python java c c++区别

大家好&#xff0c;给大家分享一下java,python,c有什么区别&#xff0c;很多人还不知道这一点。下面详细解释一下。现在让我们来看看&#xff01; 从这四种语言的难度、受欢迎度还有作用以及优点缺点给楼主做一个全面的分析&#xff0c;我们可以从中了解其区别&#xff0c;以及…

AP2400 LED汽车摩灯照明电源驱动 过EMC DC-DC降压恒流IC

产品特点 宽输入电压范围&#xff1a;5V&#xff5e;100V 可设定电流范围&#xff1a;10mA&#xff5e;6000mA 固定工作频率&#xff1a;150KHZ 内置抖频电路&#xff0c;降低对其他设备的 EMI干扰 平均电流模式采样&#xff0c;恒流精度更高 0-100%占空比控制&#xff0…

【状态估计】基于UKF法、AUKF法的电力系统三相状态估计研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

并查集练习—省份数量

上一篇中讲了并查集及其原理&#xff0c;在这篇文章中简单应用一下。如果对并查集不是很了解强烈建议先看上一篇。 题目&#xff1a; 有 n 个城市&#xff0c;其中一些彼此相连&#xff0c;另一些没有相连。如果城市 a 与城市 b 直接相连&#xff0c;且城市 b 与城市 c 直接相…

大数据Flink(五十七):Yarn集群环境(生产推荐)

文章目录 Yarn集群环境(生产推荐) 一、准备工作

目标检测中的IOU

IOU 什么是IOU?IOU应用场景写代码调试什么是IOU? 简单来说IOU就是用来度量目标检测中预测框与真实框的重叠程度。在图像分类中,有一个明确的指标准确率来衡量模型分类模型的好坏。其公式为: 这个公式显然不适合在在目标检测中使用。我们知道目标检测中都是用一个矩形框住…

【福建事业单位-推理判断】02图形推理(数量-空间重构)

【福建事业单位-推理判断】02图形推理&#xff08;数量-空间重构&#xff09; 一、数量规律1.1点&#xff08;交点、切点&#xff09;点的细化考法总结 1.2线条&#xff08;线条的数量&#xff09;线的细化考点一笔画&#xff08;重点&#xff09;一笔画的判定 总结 1.3 面面的…

flutter开发实战-video_player视频播放功能及视频缓存

flutter开发实战-video_player视频播放功能及视频缓存 最近开发过程中video_player播放视频&#xff0c; 一、引入video_player 在pubspec.yaml引入video_player video_player: ^2.7.0在iOS上&#xff0c;video_player使用的是AVPlayer进行播放。 在Android上&#xff0c;…

医疗实施-集成平台下门诊就诊流程详解

目录 集成平台下门诊就诊流程详解1.患者建档2. 门诊挂号3. 医生就诊4.处方开立5.费用收取、6、科室执行医嘱集成平台下门诊就诊流程详解 这篇文章是考虑医院使用了集成平台之后,门诊就诊流程详解。与我的文章《医疗实施-门诊就诊流程详解》的大致一样,供学有余力的人阅读。 …

AMASS database

AMASS是一个由不同的光学标记运动捕捉数据集统一表示在一个公共框架和参数化下的大型人体运动数据库。它包含了超过40小时的运动数据&#xff0c;涵盖了300多个主体和11000多个运动。它使用了SMPL人体模型&#xff0c;它是一种基于混合形状和姿态空间的生成式人体模型&#xff…

spring boot中web容器配置

web容器配置 spring boot 默认的web容器是 tomcat&#xff0c;如果需要换成其他的 web 容器&#xff0c;可以如下配置。 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><!-- 默…

springboot()—— swagger

零、一张图读懂swagger 懂了&#xff0c;这玩意就是用swagger搞出来的&#xff01; 就是一个后端开发自测的东西嘛&#xff01; 一、概念 存在即合理&#xff0c;我们看一下swagger诞生的原因&#xff1a;在前后端分离的架构中&#xff0c;前端新增一个字段&#xff0c;后端就…

Git rebase和merge区别详解

文章目录 变基的基础用法变基过程中的冲突解决冲突后无法push问题更新变基后的代码更有趣的变基用法变基的风险用变基解决变基变基 vs 合并 此文在阅读前需要有一定的git命令基础&#xff0c;若基础尚未掌握&#xff0c;建议先阅读这篇文章Git命令播报详版 在 Git 中整合来自不…

iOS 后台运行

iOS后台行&#xff0c;一般有两种方式&#xff1a; 1.UIBackgroundTaskIdentifier后台任务标记时, 2.设置后台运行模式&#xff0c;需要有voip&#xff0c;location功能的才行。不然app上线审核肯定是过不了的。 下面是我学习后台运行的尝试过程。 一.首先创建一个项目功程…

redis缓存雪崩和缓存击穿

目录 缓存雪崩 解决方案&#xff1a; 缓存击穿 ​解决方案 缓存雪崩 缓存雪崩是指在同一时段大量的缓存key同时失效或者Redis服务宕机&#xff0c;导致大量请求到达数据库&#xff0c;带来巨大压力。 解决方案&#xff1a; u 给不同的 Key 的 TTL 添加随机值 u 利用 Redis …

【力扣】23. 合并 K 个升序链表 <链表指针、堆排序、分治>

目录 【力扣】23. 合并 K 个升序链表题解方法一&#xff1a;暴力&#xff0c;先遍历取出来值到数组中排序&#xff0c;再生成新链表方法二&#xff1a;基础堆排序&#xff08;使用优先队列 PriorityQueue&#xff09;方法三&#xff1a;基础堆排序&#xff08;使用优先队列 Pri…

【深度学习笔记】深度学习框架

本专栏是网易云课堂人工智能课程《神经网络与深度学习》的学习笔记&#xff0c;视频由网易云课堂与 deeplearning.ai 联合出品&#xff0c;主讲人是吴恩达 Andrew Ng 教授。感兴趣的网友可以观看网易云课堂的视频进行深入学习&#xff0c;视频的链接如下&#xff1a; 神经网络和…

【深度学习】采用自动编码器生成新图像

一、说明 你知道什么会很酷吗&#xff1f;如果我们不需要所有这些标记的数据来训练 我们的模型。我的意思是标记和分类数据需要太多的工作。 不幸的是&#xff0c;大多数现有模型从支持向量机到卷积神经网&#xff0c;没有它们&#xff0c;卷积神经网络就无法训练。无监督学习不…