目录
2.1void Savecontact(const struct contact* pc);(保存通讯录)
2.2 void Loadcontact(struct contact* pc);(加载文件中的信息到通讯录)
前言
之前写了动态版本的通讯录,但是每次打开联系人的信息都要重新输入,这次通过文件操作写通讯录,就解决了这个问题。
1.改进方法
为了解决每次打开联系人的信息都要重新输入的问题,在原先动态版本的基础上添加了两点。
1.1保存通讯录
通过文件只写操作方式,若有文件直接打开,若没有文件新建一个文件。每次退出后,添加的联系人都会保存在文件中。
1.2加载文件中的信息到通讯录
通过文件只读操作打开文件,会加载文件中的信息到通讯录,避免了之前要重新输入的问题。
2.添加函数
2.1void Savecontact(const struct contact* pc);(保存通讯录)
2.1.1分析
1>.assert(pc); 断言pc
2>打开文件,
FILE * fopen ( const char * filename, const char * mode );
wb文件操作方式的含义是打开一个二进制文件,输入数据,若文件不存在,则新建一个文件
3>判断文件是否为空,若为空打印系统错误信息并且返回,若不为空,继续。
4>写文件(二进制形式)
fwrite函数声明
size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream)
参数
- ptr -- 这是指向要被写入的元素数组的指针。
- size -- 这是要被写入的每个元素的大小,以字节为单位。
- nmemb -- 这是元素的个数,每个元素的大小为 size 字节。
- stream -- 这是指向 FILE 对象的指针,该 FILE 对象指定了一个输出流。
将所有信息放入文件(sz-->已经放进去的信息)
指向要被写入的元素数组的指针是pc->data+i
元素是联系人的个人信息,每个元素的大小是sizeof(struct PeoInfo)
元素的个数是1
指向 FILE 对象的指针是pfw
5>关闭文件
int fclose ( FILE * stream );
pfw=NULL;(文件关闭后一定要置空)
2.1.2 代码
//保存通讯录
void Savecontact(const struct contact* pc)
{
assert(pc);
//打开文件
FILE* pfw = fopen("data.txt","wb");
if (NULL == pfw)
{
perror("Savecontact::fopen");
return;
}
//写文件-二进制形式
int i = 0;
for (int i = 0; i < pc->sz; i++)
{
fwrite(pc->data + i, sizeof(struct PeoInfo), 1, pfw);
}
//关闭文件
fclose(pfw);
pfw = NULL;
}
2.2 void Loadcontact(struct contact* pc);(加载文件中的信息到通讯录)
2.2.1分析
1>函数放到初始化通讯录函数的结尾
2>打开文件
FILE * fopen ( const char * filename, const char * mode );
rb文件操作的含义是打开二进制文件,输入数据如果文件不存在,则报错
3>判断文件是否为空,若为空打印系统错误信息并且返回,若不为空,继续。
4>读文件
4.1>创建一个空数组tmp,类型struct PeoInfo
4.2>fread声明
size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream)
从给定流 stream 读取数据到 ptr 所指向的数组中。
参数
- ptr -- 这是指向带有最小尺寸 size*nmemb 字节的内存块的指针。
- size -- 这是要读取的每个元素的大小,以字节为单位。
- nmemb -- 这是元素的个数,每个元素的大小为 size 字节。
- stream -- 这是指向 FILE 对象的指针,该 FILE 对象指定了一个输入流。
指向带有最小尺寸 size*nmemb 字节的内存块的指针是&tmp
元素是联系人的信息,每个元素的大小是sizeof(struct PeoInfo)
元素个数是1
指向 FILE 对象的指针是pfr
4.3>增加信息(因为check_capacity函数在Loadcontact函数的后面,为了能调用check_capacity函数,要在Loadcontact函数之前声明check_capacity函数)
ptr 所指向的数组中的信息,放入存放数据的空间
5>关闭文件
int fclose ( FILE * stream );
pfr=NULL;(文件关闭后一定要置空)
2.2.2 代码
static int check_capacity(struct contact* pc);
void Loadcontact(struct contact* pc)
{
//打开文件
FILE* pfr= fopen("data.txt", "rb");
if (NULL == pfr)
{
perror("Loadcontact::fopen");
return;
}
//读文件
struct PeoInfo tmp = { 0 };
while (fread(&tmp, sizeof(struct PeoInfo), 1, pfr) == 1)
{
check_capacity(pc);
pc->data[pc->sz] = tmp;
pc->sz++;
}
//关闭文件
fclose(pfr);
pfr = NULL;
}
3.完整代码
3.1test.c
#define _CRT_SECURE_NO_WARNINGS
#include"contact.h"
void meun()
{
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");
}
int main()
{
int input = 0;
struct contact con;
//初识化通讯录
Initcontact(&con);
do {
meun();
printf("请选择>\n");
scanf("%d", &input);
switch (input)
{
case 1:
Addcontact(&con);
break;
case 2:
Delcontact(&con);
break;
case 3:
Searchcontact(&con);
break;
case 4:
Modifycontact(&con);
break;
case 5:
Showcontact(&con);
break;
case 6:
Sortcontact(&con);
break;
case 0:
Savecontact(&con);
printf("文件保存成功\n");
Destroycontact(&con);
printf("退出通讯录\n");
break;
default:
printf("输入错误\n");
break;
}
} while (input);
return 0;
}
3.2contact.c
#define _CRT_SECURE_NO_WARNINGS
#include"contact.h"
static int check_capacity(struct contact* pc);
void Loadcontact(struct contact* pc)
{
//打开文件
FILE* pfr= fopen("data.txt", "rb");
if (NULL == pfr)
{
perror("Loadcontact::fopen");
return;
}
//读文件
struct PeoInfo tmp = { 0 };
while (fread(&tmp, sizeof(struct PeoInfo), 1, pfr) == 1)
{
check_capacity(pc);
pc->data[pc->sz] = tmp;
pc->sz++;
}
//关闭文件
fclose(pfr);
pfr = NULL;
}
//初始化通讯录
void Initcontact(struct contact* pc)
{
assert(pc);
pc->data = (struct PeoInfo*)malloc(DEFAULT_SZ * sizeof(struct PeoInfo));
if (pc->data == NULL)
{
perror("Initcontact()");
return;
}
pc->sz = 0;
pc->capacity = DEFAULT_SZ;
//加载文件中的信息到通讯录
Loadcontact(pc);
}
//销毁通讯录
void Destroycontact(struct contact* pc)
{
free(pc->data);
pc->data = NULL;
pc->capacity = 0;
pc->sz = 0;
}
//增加信息
static int check_capacity(struct contact* pc)
{
//空间已满
if (pc->sz == pc->capacity)
{
//增容
struct PeoInfo* ptr = (struct PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(struct PeoInfo));
if (ptr != NULL)
{
pc->data = ptr;
pc->capacity += INC_SZ;
printf("增容成功\n");
return 1;
}
else
{
perror("Addcontact()");
return 0;
}
}
else
return 1;
}
void Addcontact(struct contact* pc)
{
assert(pc);
if (0 == check_capacity(pc))
{
return;
}
//增加信息
printf("请输入名字>\n");
scanf("%s", pc->data[pc->sz].name);//字符串的名字是首地址
printf("请输入性别>\n");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄>\n");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入电话>\n");
scanf("%s", pc->data[pc->sz].tele);
printf("请输入地址>\n");
scanf("%s", pc->data[pc->sz].addr);
pc->sz++;
printf("添加成功\n");
}
//显示信息
void Showcontact(const struct contact* pc)
{
printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "名字", "性别", "年龄", "电话", "地址");
for (int i = 0; i < pc->sz; i++)
{
printf("%-20s\t%-5s\t%-5d\t%-12s\t%-30s\n", pc->data[i].name, pc->data[i].sex, pc->data[i].age, pc->data[i].tele, pc->data[i].addr);
}
}
int Find(const struct contact* pc, char name[])
{
for (int i = 0; i < pc->sz; i++)
{
if (0 == strcmp(pc->data[i].name, name));
return i;
}
return -1;
}
//删除联系人
void Delcontact(struct contact* pc)
{
char name[MAX_name];
printf("请输入要删除的联系人的名字\n");
scanf("%s", name);
int ret = Find(pc, name);
//查找指定人是否存在
if (ret == -1)
{
printf("联系人不存在\n");
return;
}
for (int j = 0; j < pc->sz - 1; j++)
{
pc->data[j] = pc->data[j + 1];
}
pc->sz--;
printf("删除成功\n");
}
//查找指定联系人的信息
void Searchcontact(const struct contact* pc)
{
char name[MAX_name];
printf("请输入要查找的人的名字:>");
scanf("%s", name);
//查找一下指定的人是否存在
int ret = Find(pc, name);
if (ret == -1)
printf("要查找的人不存在\n");
else
{
printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "性别", "年龄", "电话", "地址");
printf("%-20s\t%-5s\t%-5d\t%-12s\t%-30s\n", pc->data[ret].name,
pc->data[ret].sex,
pc->data[ret].age,
pc->data[ret].tele,
pc->data[ret].addr);
}
}
//修改
void Modifycontact(struct contact* pc)
{
char name[MAX_name];
printf("请输入要查找人的名字\n");
scanf("%s", name);
//查找指定人是否存在
int ret = Find(pc, name);
if (ret == -1)
{
printf("要修改的人不存在\n");
return;
}
//修改信息
printf("请输入名字>\n");
scanf("%s", pc->data[pc->sz].name);//字符串的名字是首地址
printf("请输入性别>\n");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄>\n");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入电话>\n");
scanf("%s", pc->data[pc->sz].tele);
printf("请输入地址>\n");
scanf("%s", pc->data[pc->sz].addr);
pc->sz++;
printf("添加成功\n");
}
//排序
void CmpByName(const void* e1, const void* e2)
{
return strcmp(((struct PeoInfo*)e1)->name, ((struct PeoInfo*)e2)->name);
}
void CmpByAge(const void* e1, const void* e2)
{
return ((struct PeoInfo*)e1)->age - ((struct PeoInfo*)e2)->age;
}
void Sortcontact(struct contact* pc)
{
按名字排
//qsort(pc->data, pc->sz, sizeof(struct PeoInfo), CmpByName);
//按年龄排
qsort(pc->data, pc->sz, sizeof(struct PeoInfo), CmpByAge);
}
//保存通讯录
void Savecontact(const struct contact* pc)
{
assert(pc);
//打开文件
FILE* pfw = fopen("data.txt","wb");
if (NULL == pfw)
{
perror("Savecontact::fopen");
return;
}
//写文件-二进制形式
int i = 0;
for (int i = 0; i < pc->sz; i++)
{
fwrite(pc->data + i, sizeof(struct PeoInfo), 1, pfw);
}
//关闭文件
fclose(pfw);
pfw = NULL;
}
3.3contact.h
#pragma once
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<string.h>
#define MAX 100
#define MAX_name 20
#define MAX_sex 5
#define MAX_tele 12
#define MAX_addr 30
//通讯录默认大小
#define DEFAULT_SZ 2
//每次增加2个
#define INC_SZ 2
//表示一个人的信息
struct PeoInfo
{
char name[MAX_name];
char sex[MAX_sex];
int age;
char tele[MAX_tele];
char addr[MAX_addr];
};
//通讯录
struct contact
{
struct PeoInfo* data;//指向存放数据的空间
int sz;//已经放进去的信息
int capacity;//容量
};
//初始化通讯录
void Initcontact(struct contact* pc);
//增加信息
void Addcontact(struct contact* pc);
//显示联系人
void Showcontact(const struct contact* pc);
//删除联系人
void Delcontact(struct contact* pc);
//查找联系人
void Searchcontact(const struct contact* pc);
//修改
void Modifycontact(struct contact* pc);
//排序
void Sortcontact(struct contact* pc);
//销毁通讯录
void Destroycontact(struct contact* pc);
//保存通讯录
void Savecontact(const struct contact* pc);
//加载文件中的信息到通讯录
void Loadcontact(struct contact* pc);