C语言改造通讯录操作详解

 更新时间:2023年01月17日 10:51:47   作者:戊子仲秋  
这篇文章主要介绍了C语言文件操作改造通讯录方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧

文件改造通讯录需要修改的地方:

1.在通讯录退出前写入文件

在contact.c文件中实现:

//写入文件(保存通讯录)
void SaveContact(const Contact* pc)
{
	//写数据
	FILE* pf = fopen("contact.txt", "wb");
	//判断写入是否成功
	if (pf == NULL)
	{
		perror("SaveContact::fopen");
	}
	else
	{
		int i = 0;
		for (i = 0; i < pc->sz; i++)
		{
			//遍历数组,将数组每个元素写入文件
			fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);
		}
		//关闭文件
		fclose(pf);
		pf = NULL;
		printf("保存成功\n");
	}
}

在contact.h文件中引用:

//保存通讯录中的信息到文件中
void SaveContact(const Contact* pc);

2.改造初始化通讯录

在contact.c文件中实现:

//初始化通讯录//动态版//文件版
void InitContact(Contact* pc)
{
	assert(pc);
	pc->sz = 0;//通讯录中存放0个人的信息
	PeoInfo* ptr = (PeoInfo*)calloc(DEFAULT_SZ, sizeof(PeoInfo));//通过calloc开辟空间
	//判断
	if (ptr == NULL)
	{
		perror("InitContact::calloc");
		return;
	}
	pc->data = ptr;//data指针得到开辟的空间的地址
	pc->capacity = DEFAULT_SZ;//初始容量赋值
	//添加:
	//加载文件信息到通讯录
	LoadContact(pc);
}

在通讯录初始化的时候加载文件的信息。

分装一个加载函数实现:

//加载文件
void LoadContact(Contact* pc)
{
	//打开文件
	FILE* pf = fopen("contact.txt", "rb");
	//判断打开是否成功
	if (pf == NULL)
	{
		perror("LoadContact::fopen");
	}
	else
	{
		//创建一个空的指针变量
		PeoInfo tmp = { 0 };
		int i = 0;
		//循环读取文件信息
		while (fread(&tmp, sizeof(PeoInfo), 1, pf))//当文件返回0时跳出循环
		{
			//每次读取前调用增容函数,判断是否需要增容
			CheckCapacity(pc);
			//通讯录数组接收文件信息
			pc->data[i] = tmp;
			//通讯录存放人数++
			pc->sz++;
			i++;
		}
		//关闭文件
		fclose(pf);
		pf = NULL;
	}
}

这样,文件改造的通讯录就完成了。

以下是通讯录终极版本的源码。

3.通讯录源码

test.c文件:

#define _CRT_SECURE_NO_WARNINGS 1
#include "contact.h"
void menu()
{
	printf("\n");
	printf("—————————— 通讯录 ———————————-\n");
	printf("—————————————————————————-\n");
	printf("————————   1.添加联系人    ————————\n");
	printf("—————————————————————————-\n");
	printf("————————   2.删除联系人    ————————\n");
	printf("—————————————————————————-\n");
	printf("————————   3.查找联系人    ————————\n");
	printf("—————————————————————————-\n");
	printf("————————   4.修改联系人信息  ———————\n");
	printf("—————————————————————————-\n");
	printf("————————   5.整理通讯录    ————————\n");
	printf("—————————————————————————-\n");
	printf("————————   6.查看整个通讯录  ———————\n");
	printf("—————————————————————————-\n");
	printf("————————   0.保存并退出    ————————\n");
	printf("—————————————————————————-\n");
	printf("\n");
}
void test()
{
	int input = 0;
	//创建通讯录con
	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 SORT:
			SortContact(&con);
			break;
		case SHOW:
			ShowContact(&con);
			break;
		case EXIT:
			//DestroyContact(&con);
			//保存文件信息
			SaveContact(&con);
			printf("通讯录已退出\n");
			break;
		default:
			printf("选择错误\n");
			break;
		}
	} while (input);
}
int main()//主函数里不要放太多东西
{
	test();
	return 0;
}

contact.h文件:

#pragma once//防止头文件重复引用
//提前将需要使用的头文件引用,
//具体实现通讯录是,需要头文件可以直接添加
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>
//通过#define定义的常量,方便管理和使用
#define MAX 100
#define NAME_MAX 20
#define SEX_MAX 5
#define ADDR_MAX 30
#define TELE_MAX 12
//通讯录初始的大小和每次增容的大小
#define DEFAULT_SZ 3
#define INC_SZ 2
//通讯录中存放一个人的信息
typedef struct PeoInfo//typedef简化结构体名称
{
	char name[NAME_MAX];
	int age;
	char sex[SEX_MAX];
	char addr[ADDR_MAX];
	char tele[TELE_MAX];
}PeoInfo;
//动态增长版本的通讯录
typedef struct Contact
{
	PeoInfo* data;//指向存放人信息的空间
	int sz;//用来存放数组元素个数
	int capacity;//当前通讯录的最大容量
}Contact;
创建一个结构体将数组和数组中存放的元素数封装//静态版本
//typedef struct Contact
//{
//	PeoInfo data[MAX];//data这个数组元素的类型的是结构体PeoInfo
//	                  //用来存放联系人信息
//	int sz;//用来存放数组元素个数
//}Contact;
enum Option
{
	EXIT,//0
	ADD,
	DEL,
	SEARCH,
	MODIFY,
	SORT,
	SHOW
};
//初始化通讯录
void InitContact(Contact* pc);
销毁通讯录
//void DestroyContact(Contact* pc);
//增加联系人
void AddContact(Contact* pc);
//删除联系人
void DelContact(Contact* pc);
//查找联系人
void SearchContact(const Contact* pc);
//修改指定联系人
void ModifyContact(Contact* pc);
//整理通讯录
void SortContact(Contact* pc);
//显示通讯录的信息
void ShowContact(const Contact* pc);
//保存通讯录中的信息到文件中
void SaveContact(const Contact* pc);
//加载文件信息到通讯录
void LoadContact(Contact* pc);

contact.c文件:

#define _CRT_SECURE_NO_WARNINGS 1
#include "contact.h"
初始化通讯录//静态版
//void InitContact(Contact* pc)
//{
//  assert(pc);
//	pc->sz = 0;//代表数组中有0个元素
//	memset(pc->data, 0, sizeof(pc->data));//data是整个数组的大小,初始化成0
//}
//初始化通讯录//动态版//文件版
void InitContact(Contact* pc)
{
	assert(pc);
	pc->sz = 0;//通讯录中存放0个人的信息
	PeoInfo* ptr = (PeoInfo*)calloc(DEFAULT_SZ, sizeof(PeoInfo));//通过calloc开辟空间
	//判断
	if (ptr == NULL)
	{
		perror("InitContact::calloc");
		return;
	}
	pc->data = ptr;//data指针得到开辟的空间的地址
	pc->capacity = DEFAULT_SZ;//初始容量赋值
	//添加:
	//加载文件信息到通讯录
	LoadContact(pc);
}
销毁创建的内存
//void DestroyContact(Contact* pc)
//{
//	free(pc->data);
//	pc->data = NULL;
//}
//增容函数
void CheckCapacity(Contact* pc)
{
	if (pc->sz == pc->capacity)//如果容量满了,就进来
	{
		//通过realloc函数进行增容,原容量+INC_SZ(可以根据自己喜好设置)
		PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(PeoInfo));
		//判断
		if (ptr == NULL)
		{
			perror("CheckCapacity::realloc");
			return;
		}
		pc->data = ptr;//data指针接收增容后的内存的地址
		pc->capacity += INC_SZ;//容量也按设定增加
		printf("增容成功\n");//提示增容成效
	}
}
增加联系人//静态版
//void AddContact(Contact* pc)
//{
//	assert(pc);
//	if (pc->sz == MAX)//如果通讯录满了
//	{
//		printf("通讯录已满,无法添加\n");
//		return;//就会直接返回
//	}
//
//	//增加一个人的信息
//	printf("请输入名字:>");
//	scanf("%s", pc->data[pc->sz].name);
//	//通过pc指针访问data数组的结构体类型的元素,进而访问结构体成员
//
//	printf("请输入年龄:>");
//	scanf("%d", &(pc->data[pc->sz].age));//age不是数组,需要取地址
//
//	printf("请输入性别:>");
//	scanf("%s", pc->data[pc->sz].sex);
//
//	printf("请输入地址:>");
//	scanf("%s", pc->data[pc->sz].addr);
//
//	printf("请输入电话:>");
//	scanf("%s", pc->data[pc->sz].tele);
//
//	pc->sz++;//代表数组中的元素个数+1
//}
//增加联系人//动态版
void AddContact(Contact* pc)
{
	assert(pc);
	//增容
	CheckCapacity(pc);
	//增加一个人的信息
	printf("请输入名字:>");
	scanf("%s", pc->data[pc->sz].name);
	//通过pc指针访问data数组的结构体类型的元素,进而访问结构体成员
	printf("请输入年龄:>");
	scanf("%d", &(pc->data[pc->sz].age));//age不是数组,需要取地址
	printf("请输入性别:>");
	scanf("%s", pc->data[pc->sz].sex);
	printf("请输入地址:>");
	scanf("%s", pc->data[pc->sz].addr);
	printf("请输入电话:>");
	scanf("%s", pc->data[pc->sz].tele);
	pc->sz++;//代表数组中的元素个数+1
}
//显示通讯录的信息
void ShowContact(const Contact* pc)
{
	assert(pc);
	printf("%-10s\t%-4s\t%-5s\t%-20s\t%-12s\n", "名字", "年龄", "性别", "地址", "电话");
	int i = 0;
	for (i = 0; i < pc->sz; i++)//遍历通讯录并打印
	{
		printf("%-10s\t%-4d\t%-5s\t%-20s\t%-12s\n",
				pc->data[i].name,
 				pc->data[i].age,
				pc->data[i].sex,
				pc->data[i].addr,
				pc->data[i].tele);
	}
}
//查找函数
int FindByName(const Contact* pc, char name[])
{
	int i = 0;
	int del = 0;
	for (i = 0; i < pc->sz; i++)//遍历通讯录
	{
		if (strcmp(pc->data[i].name, name) == 0)
		{   //通过strcmp函数判断要查找的联系人是否存在
			del = i;
			return del;//返回数组下标(要查找的元素的位置)
		}
	}
	return -1;//找不到
}
//删除联系人
void DelContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX] = { 0 };//初始化name数组(字符串)
	if (pc->sz == 0)//判断通讯录中是否存在联系人
	{
		printf("通讯录为空,无法删除\n");
		return;
	}
	//找到要删除的人
	printf("请输入要删除的人的名字:>");
	scanf("%s", name);//输入字符串
	int ret = FindByName(pc, name);//分装字符串查找函数
	if (-1 == ret)
	{
		printf("要删除的人不存在\n");
		return;
	}
	//删除
	int i = 0;
	for (i = ret; i < pc->sz - 1; i++)
	{
		pc->data[i] = pc->data[i + 1];//将存放在被删除的联系人后面的联系人信息,
   	}                                 //通过循环一个个往前覆盖
	pc->sz--;//数组元素-1
	printf("删除成功\n");
}
//查找联系人
void SearchContact(const Contact* pc)
{
	assert(pc);
	char name[NAME_MAX] = { 0 };//初始化name数组(字符串)
	printf("请输入要查找人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);//函数复用
	if (-1 == pos)
	{
		printf("要查找的人不存在\n");
		return;
	}
	//打印信息//我实现的是左对齐,并用水平制表符使打印出来的观感更好
	printf("%-10s\t%-4s\t%-5s\t%-20s\t%-12s\n", "名字", "年龄", "性别", "地址", "电话");
		printf("%-10s\t%-4d\t%-5s\t%-20s\t%-12s\n",
			pc->data[pos].name,
			pc->data[pos].age,
			pc->data[pos].sex,
			pc->data[pos].addr,
			pc->data[pos].tele);
}
//修改指定联系人
void ModifyContact(Contact* pc)
{
	assert(pc);
	char name[NAME_MAX] = { 0 };//初始化name数组(字符串)
	printf("请输入要修改人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);//函数复用
	if (-1 == pos)
	{
		printf("要修改的人不存在\n");
		return;
	}
	//重新录入
	printf("请输入修改后的名字:>");
	scanf("%s", pc->data[pos].name);
	printf("请输入修改后的年龄:>");
	scanf("%d", &(pc->data[pos].age));//age不是数组,需要取地址
	printf("请输入修改后的性别:>");
	scanf("%s", pc->data[pos].sex);
	printf("请输入修改后的地址:>");
	scanf("%s", pc->data[pos].addr);
	printf("请输入修改后的电话:>");
	scanf("%s", pc->data[pos].tele);
	printf("修改完成\n");
}
//整形排序的实现
int CmpContactByAge(const void* e1, const void* e2)
{
	//这个排出来的是升序,如果想排降序,只需将e1和e2的位置调换即可
	return ((Contact*)e1)->data->age - ((Contact*)e2)->data->age;
}
//整理通讯录
void SortContact(Contact* pc)
{
	assert(pc);
	int sz = pc->sz;
	//通过qsort函数辅助排序
	qsort(pc->data, sz, sizeof(pc->data[0]), CmpContactByAge);
	printf("排序成功\n");
}
//写入文件(保存通讯录)
void SaveContact(const Contact* pc)
{
	//写数据
	FILE* pf = fopen("contact.txt", "wb");
	//判断写入是否成功
	if (pf == NULL)
	{
		perror("SaveContact::fopen");
	}
	else
	{
		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 LoadContact(Contact* pc)
{
	//打开文件
	FILE* pf = fopen("contact.txt", "rb");
	//判断打开是否成功
	if (pf == NULL)
	{
		perror("LoadContact::fopen");
	}
	else
	{
		//创建一个空的指针变量
		PeoInfo tmp = { 0 };
		int i = 0;
		//循环读取文件信息
		while (fread(&tmp, sizeof(PeoInfo), 1, pf))//当文件返回0时跳出循环
		{
			//每次读取前调用增容函数,判断是否需要增容
			CheckCapacity(pc);
			//通讯录数组接收文件信息
			pc->data[i] = tmp;
			//通讯录存放人数++
			pc->sz++;
			i++;
		}
		//关闭文件
		fclose(pf);
		pf = NULL;
	}
}

到此这篇关于C语言改造通讯录操作详解的文章就介绍到这了,更多相关C语言改造通讯录内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++冒泡排序算法实例

    C++冒泡排序算法实例

    这篇文章主要介绍了C++冒泡排序算法实例,本文先是介绍了什么是冒泡排序,然后给出了实现代码,需要的朋友可以参考下
    2014-10-10
  • 从头学习C语言之字符串处理函数

    从头学习C语言之字符串处理函数

    这篇文章主要为大家详细介绍了C语言之字符串处理函数,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下, 希望能够给你带来帮助
    2022-01-01
  • C语言中数据是如何存储在内存中的

    C语言中数据是如何存储在内存中的

    使用编程语言进行编程时,需要用到各种变量来存储各种信息。变量保留的是它所存储的值的内存位置。这意味着,当您创建一个变量时,就会在内存中保留一些空间。您可能需要存储各种数据类型的信息,操作系统会根据变量的数据类型,来分配内存和决定在保留内存中存储什么
    2022-04-04
  • C++ template用法案例详解

    C++ template用法案例详解

    这篇文章主要介绍了C++ template用法案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-09-09
  • C++实践分数类中运算符重载的方法参考

    C++实践分数类中运算符重载的方法参考

    今天小编就为大家分享一篇关于C++实践分数类中运算符重载的方法参考,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-02-02
  • 浅析C++中static的一些用法

    浅析C++中static的一些用法

    static是静止的,静态的意思,那它有什么用呢,今天通过实例代码讲解下C++中static的一些用法,感兴趣的朋友跟随小编一起看看吧
    2022-12-12
  • 详解C语言中双指针算法的使用

    详解C语言中双指针算法的使用

    双指针,指的是在遍历对象的过程中,不是普通的使用单个指针进行访问,而是使用两个相同方向(快慢指针)或者相反方向(对撞指针)的指针进行扫描,从而达到相应的目的。本文将通过示例带大家深入了解双指针算法的使用
    2022-08-08
  • C语言之结构体定义 typedef struct 用法详解和用法小结

    C语言之结构体定义 typedef struct 用法详解和用法小结

    这篇文章主要介绍了C语言的结构体定义typedef struct用法详解和用法小结,typedef是类型定义,typedef struct 是为了使用这个结构体方便,感兴趣的同学可以参考阅读
    2023-03-03
  • Recommended C Style and Coding Standards中文翻译版

    Recommended C Style and Coding Standards中文翻译版

    本文翻译自Recommended C Style and Coding Standards(C语言编码风格和标准),需要的朋友可以参考下
    2014-04-04
  • c++中的stack和dequeue解析

    c++中的stack和dequeue解析

    这篇文章主要介绍了c++中的stack和dequeue介绍,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-05-05

最新评论