目录
我们发现静态开辟的通讯录名单固定为1000,少了浪费空间,多了又要重新修改,我们能不能利用动态内存开辟实现初始化通讯录?
我们退出程序后通讯录便被销毁了,我们又能否把我们所输入的名单保存至文件中,便于添加和查看?
为什么说通讯录是迈向数据结构呢?原因有两点:
1.这篇博客写完我们进要进入到数据结构学习
2.通讯录是对学习C语言的总结,建议一定要自己动手写出通讯录,因为用数据结构实现C语言需要基础扎实牢固,写出通讯录代表知识掌握的不错
为了规范代码,模块化书写代码,我们采用多文件编译
主函数的实现 | test.c |
函数声明 | Contact.h |
函数实现 | Contact.c |
静态版本实现通讯录:(数组大小不可修改)
1.创建一个通讯录结构体,结构体内部包含姓名,性别,电话号码,年龄,还需要一个记录通讯录中已经保存的信息个数(让增删查改时便于修改通讯录名单)
2.头文件包含应该在头文件中,防止多次包含头文件
3.实现main函数时应该把main函数主体构造出来,为了规范代码,我们可以创建一个枚举类型,通过枚举类型来书写代码
test.c主体构造
enum contact
{
EXIT,//0
ADD,
DEL,
SEARCH,
MODIFY,
SORT,
PRINT,
};
void menu()
{
printf("*****************************\n");
printf("**** 1.ADD 2.DEL ********\n");
printf("**** 3.SEARCH 4.MODIFY ******\n");
printf("**** 5.SORT 6.PRINT ********\n");
printf("*****0.EXIT ********\n");
printf("*****************************\n");
}
void test()
{
int input = 0;
do
{
menu();
printf("请选择->");
scanf("%d", &input);
switch (input)
{
case EXIT:
break;
case ADD:
break;
case DEL:
case SEARCH:
break;
case MODIFY:
break;
case SORT:
break;
case PRINT:
break;
}
} while (input);
}
int main()
{
test();
return 0;
}
4.头文件的实现,我们可以把刚刚的枚举放到头文件中,其中通过#define作用是方便修改,如果我们想扩大缩小通讯录名单,例如姓名数组大小时十分方便,只需要改define即可
Contact.h主体构造
//通讯录实现 接口申明
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>
#define MAX 1000
#define NAME 20
#define SEX 5
#define PHONE 12
#define ADDR 30
#define DEFAULF_SZ 3
typedef struct PeoInfo
{
char name[NAME];//姓名
char sex[SEX];//性别
int age;//年龄
char phone[PHONE];//电话
char addr[ADDR];//地址
}PeoInfo;
enum contact
{
EXIT,
ADD,
DEL,
SEARCH,
MODIFY,
SORT,
PRINT,
};
静态版本
typedef struct Contact
{
PeoInfo data[MAX];
int sz;
}Contact;
声明我们所需的函数
// 初始化通讯录
void InitContact(Contact* pc);
//销毁通讯录
void DestoryContact(Contact* pc);
//添加通讯录名单
void Addcontact(Contact * pc);
//打印通讯录名单
void PrintContact(const Contact* pc);
//删除通讯录名单
void DelContact(Contact* pc);
//查找通讯录名单
void SearchContact(const Contact* pc);
//改通讯录名单
void ModifyContact(Contact* pc);
//保存通讯录到文件
void SaveContact(const Contact* pc);
5.如何实现通讯录所需的函数,我们首先需要创建通讯录后,初始化通讯录(在调用main函数时创建后便初始化)
Contact.c函数实现初始化通讯录,先把数组内容初始化为0
静态版本
void InitContact(Contact* pc)//初始化通讯录
{
assert(pc);
memset(pc->data,0,sizeof(pc->data));
pc->sz = 0;
}
Contact.c函数实现添加通讯录名单,如果数组sz下标已经满了=1000,停止放入
void Addcontact(Contact* pc)
{
assert(pc);
if (pc->sz == MAX)
{
printf("通讯录已满,无法添加\n");
return;
}
printf("请输入姓名");
scanf("%s", pc->data[pc->sz].name);
printf("请输入性别");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入手机号");
scanf("%s", pc->data[pc->sz].phone);
printf("请输入地址");
scanf("%s", pc->data[pc->sz].addr);
pc->sz++;
printf("添加成功!\n");
}
Contact.c函数实现查找通讯录名单,输入姓名进新数组中,遍历查找,找到了返回数组sz下标
int FindContact(const Contact* pc,char name[])
{
assert(pc);
int i = 0;
for (i = 0; i < pc->sz; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
return i;
}
}
return -1;
}
Contact.c函数实现删除通讯录名单,输入姓名进新数组中,遍历查找,最后一个元素虽然没有被覆盖,但是sz--后找不到那里去
void DelContact(Contact* pc)
{
assert(pc);
if (pc->sz == 0)
{
printf("通讯录已空,无法删除\n");
return;
}
//找到要删除的人
printf("请输入要删除人的信息\n");
char name[NAME] = {0};
scanf("%s", name);
int pos=FindContact(pc,name);
if (pos == -1)
{
printf("找不到\n");
return;
}
int j = 0;
for (j = pos; j < pc->sz-1; j++)
{
pc->data[j] = pc->data[j + 1];
}
pc->sz--;
printf("删除成功\n");
}
Contact.c函数实现打印通讯录名单
void SearchContact(const Contact* pc)
{
char name[NAME] = { 0 };
printf("请输入要查找人的名字");
scanf("%s", name);
int pos = FindContact(pc, name);
if (pos == -1)
{
printf("找不到\n");
return;
}
printf("%-20s %-5s %-3s %-12s %-30s\n", "姓名", "性别", "年龄", "手机号", "地址");
printf("%-20s %-5s %-3d %-12s %-30s\n", pc->data[pos].name, pc->data[pos].sex, pc->data[pos].age, pc->data[pos].phone, pc->data[pos].addr);
}
Contact.c函数实现修改通讯录名单
void ModifyContact(Contact* pc)
{
char name[NAME] = { 0 };
printf("请输入要查找人的名字");
scanf("%s", name);
int pos = FindContact(pc, name);
if (pos == -1)
{
printf("找不到\n");
return;
}
else
{
printf("请输入姓名");
scanf("%s", pc->data[pc->sz].name);
printf("请输入性别");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入手机号");
scanf("%s", pc->data[pc->sz].phone);
printf("请输入地址");
scanf("%s", pc->data[pc->sz].addr);
printf("修改成功!");
}
}
动态版本实现通讯录
我们发现静态开辟的通讯录名单固定为1000,少了浪费空间,多了又要重新修改,我们能不能利用动态内存开辟实现初始化通讯录?
//动态版本
void InitContact(Contact* pc)
{
assert(pc);
//memset(pc->data,0,sizeof(pc->data));
pc->sz = 0;
pc->capacity = DEFAULF_SZ;
pc->data = (PeoInfo*)malloc(pc->capacity * sizeof(PeoInfo));
if (pc->data == NULL)
{
perror("InitContact:malloc");
return;
}
memset(pc->data,0, pc->capacity * sizeof(PeoInfo));
}
增容通讯录
void CheckCapacity(Contact* pc)
{
if (pc->capacity == pc->sz)
{
//增容
PeoInfo* tmp = (PeoInfo*)realloc(pc->data, (pc->capacity + 2) * sizeof(PeoInfo));//realloc调整后的总大小
if (tmp != NULL)
{
pc->data = tmp;
}
else
{
perror("CheckCapacity::realloc");
return;
}
pc->capacity += 2;
printf("增容成功\n");
}
}
既然我们利用动态内存开辟初始化好了通讯录,我们在增添通讯录名单的时候还需要检查通讯录是否满了,满了便增容
文件版本实现通讯录
我们退出程序后通讯录便被销毁了,我们又能否把我们所输入的名单保存至文件中,便于添加和查看?
文件打开后需要销毁通讯录,把在程序中所写的通讯录保存至文件中
void DestoryContact(Contact* pc)
{
free(pc->data);
pc->data = NULL;
pc->capacity = 0;
pc->sz = 0;
printf("销毁成功\n");
}
void SaveContact(const Contact* pc)
{
FILE* pf =fopen("Contact.txt", "wb");//二进制方式写进去
if (pf == NULL)
{
perror("SaveContact::fopen");
return ;
}
//写文件
int i = 0;
for (i = 0; i < pc->sz; i++)
{
fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);
}
printf("保存成功!\n");
//关闭文件
fclose(pf);
pf = NULL;
}
初始化通讯录 --文件版本
把我们之前结束时保存的通讯录名单导入加载到程序中(初始化时)
//初始化通讯录 --文件版本
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))//返回的数值比要求读写的count数据小,代表读写完成,如果返回0代表没读取到
{
//通讯录初始化也要扩容,防止数据大于初始值
CheckCapacity(pc);
pc->data[pc -> sz] = tmp;//读取一个信息放到sz中,因为默认sz=0
pc->sz++;
}
//关闭文件
fclose(pf);
pf = NULL;
}
void InitContact(Contact* pc)
{
assert(pc);
pc->sz = 0;
pc->capacity = DEFAULF_SZ;
pc->data = (PeoInfo*)malloc(pc->capacity * sizeof(PeoInfo));
if (pc->data == NULL)
{
perror("InitContact:malloc");
return;
}
memset(pc->data, 0, pc->capacity * sizeof(PeoInfo));
//加载文件信息到通讯录中
LoadContact(pc);
}
完整Contact.h头文件实现声明
//通讯录实现 接口申明
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>
#define MAX 10
#define NAME 20
#define SEX 5
#define PHONE 12
#define ADDR 30
#define DEFAULF_SZ 3
typedef struct PeoInfo
{
char name[NAME];
char sex[SEX];
int age;
char phone[PHONE];
char addr[ADDR];
}PeoInfo;
enum contact
{
EXIT,
ADD,
DEL,
SEARCH,
MODIFY,
SORT,
PRINT,
};
//静态版本
//typedef struct Contact
//{
// PeoInfo data[MAX];
// int sz;
//}Contact;
//动态版本
typedef struct Contact
{
PeoInfo *data;//可以存放动态人数
int sz;//记录通讯录中已经保存的信息个数
int capacity;//记录通讯录当前的最大容量
}Contact;
// 初始化通讯录
void InitContact(Contact* pc);
//销毁通讯录
void DestoryContact(Contact* pc);
//添加通讯录名单
void Addcontact(Contact * pc);
//打印通讯录名单
void PrintContact(const Contact* pc);
//删除通讯录名单
void DelContact(Contact* pc);
//查找通讯录名单
void SearchContact(const Contact* pc);
//改通讯录名单
void ModifyContact(Contact* pc);
//保存通讯录到文件
void SaveContact(const Contact* pc);
完整test.c源文件实现框架
//专门测试通讯录功能
#include "contact.h"
void menu()
{
printf("*****************************\n");
printf("**** 1.ADD 2.DEL ********\n");
printf("**** 3.SEARCH 4.MODIFY ******\n");
printf("**** 5.SORT 6.PRINT ********\n");
printf("*****0.EXIT ********\n");
printf("*****************************\n");
}
void test()
{
Contact con;//创建通讯录
InitContact(&con);//初始化通讯录
int input = 0;
do
{
menu();
printf("请选择->");
scanf("%d", &input);
switch (input)
{
case EXIT:
SaveContact(&con);
DestoryContact(&con);
break;
case ADD:
Addcontact(&con);
break;
case DEL:
DelContact(&con);
case SEARCH:
SearchContact(&con);
break;
case MODIFY:
ModifyContact(&con);
break;
case SORT:
break;
case PRINT:
PrintContact(&con);
break;
}
} while (input);
}
int main()
{
test();
return 0;
}
完整Contact.c源文件实现函数
//通讯录实现 接口(函数)实现
#include "contact.h"
//静态版本
//void InitContact(Contact* pc)
//{
// assert(pc);
// memset(pc->data,0,sizeof(pc->data));
// pc->sz = 0;
//}
//动态版本
//void InitContact(Contact* pc)
//{
// assert(pc);
// //memset(pc->data,0,sizeof(pc->data));
// pc->sz = 0;
// pc->capacity = DEFAULF_SZ;
// pc->data = (PeoInfo*)malloc(pc->capacity * sizeof(PeoInfo));
// if (pc->data == NULL)
// {
// perror("InitContact:malloc");
// return;
// }
// memset(pc->data,0, pc->capacity * sizeof(PeoInfo));
//}
//
//增容通讯录
void CheckCapacity(Contact* pc)
{
if (pc->capacity == pc->sz)
{
//增容
PeoInfo* tmp = (PeoInfo*)realloc(pc->data, (pc->capacity + 2) * sizeof(PeoInfo));//realloc调整后的总大小
if (tmp != NULL)
{
pc->data = tmp;
}
else
{
perror("CheckCapacity::realloc");
return;
}
pc->capacity += 2;
printf("增容成功\n");
}
}
//初始化通讯录 --文件版本
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))//返回的数值比要求读写的count数据小,代表读写完成,如果返回0代表没读取到
{
//通讯录初始化也要扩容,防止数据大于初始值
CheckCapacity(pc);
pc->data[pc -> sz] = tmp;//读取一个信息放到sz中,因为默认sz=0
pc->sz++;
}
//关闭文件
fclose(pf);
pf = NULL;
}
void InitContact(Contact* pc)
{
assert(pc);
pc->sz = 0;
pc->capacity = DEFAULF_SZ;
pc->data = (PeoInfo*)malloc(pc->capacity * sizeof(PeoInfo));
if (pc->data == NULL)
{
perror("InitContact:malloc");
return;
}
memset(pc->data, 0, pc->capacity * sizeof(PeoInfo));
//加载文件信息到通讯录中
LoadContact(pc);
}
void Addcontact(Contact* pc)
{
assert(pc);
//静态版本
/*if (pc->sz == MAX)
{
printf("通讯录已满,无法添加\n");
return;
}*/
//动态版本
CheckCapacity(pc);
//录入信息
printf("请输入姓名");
scanf("%s", pc->data[pc->sz].name);
printf("请输入性别");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入手机号");
scanf("%s", pc->data[pc->sz].phone);
printf("请输入地址");
scanf("%s", pc->data[pc->sz].addr);
pc->sz++;
printf("添加成功!\n");
}
void PrintContact(const Contact* pc)
{
assert(pc);
int i = 0;
printf("%-20s %-5s %-3s %-12s %-30s\n", "姓名", "性别", "年龄", "手机号", "地址");
for (i = 0; i < pc->sz; i++)
{
printf("%-20s %-5s %-3d %-12s %-30s\n", pc->data[i].name, pc->data[i].sex, pc->data[i].age, pc->data[i].phone, pc->data[i].addr);
}
}
int FindContact(const Contact* pc,char name[])
{
assert(pc);
int i = 0;
for (i = 0; i < pc->sz; i++)
{
if (strcmp(pc->data[i].name, name) == 0)
{
return i;
}
}
return -1;
}
void DelContact(Contact* pc)
{
assert(pc);
if (pc->sz == 0)
{
printf("通讯录已空,无法删除\n");
return;
}
//找到要删除的人
printf("请输入要删除人的信息\n");
char name[NAME] = {0};
scanf("%s", name);
int pos=FindContact(pc,name);
if (pos == -1)
{
printf("找不到\n");
return;
}
int j = 0;
for (j = pos; j < pc->sz-1; j++)
{
pc->data[j] = pc->data[j + 1];
}
pc->sz--;
printf("删除成功\n");
}
void SearchContact(const Contact* pc)
{
char name[NAME] = { 0 };
printf("请输入要查找人的名字");
scanf("%s", name);
int pos = FindContact(pc, name);
if (pos == -1)
{
printf("找不到\n");
return;
}
printf("%-20s %-5s %-3s %-12s %-30s\n", "姓名", "性别", "年龄", "手机号", "地址");
printf("%-20s %-5s %-3d %-12s %-30s\n", pc->data[pos].name, pc->data[pos].sex, pc->data[pos].age, pc->data[pos].phone, pc->data[pos].addr);
}
void ModifyContact(Contact* pc)
{
char name[NAME] = { 0 };
printf("请输入要查找人的名字");
scanf("%s", name);
int pos = FindContact(pc, name);
if (pos == -1)
{
printf("找不到\n");
return;
}
else
{
printf("请输入姓名");
scanf("%s", pc->data[pc->sz].name);
printf("请输入性别");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入手机号");
scanf("%s", pc->data[pc->sz].phone);
printf("请输入地址");
scanf("%s", pc->data[pc->sz].addr);
printf("修改成功!");
}
}
void DestoryContact(Contact* pc)
{
free(pc->data);
pc->data = NULL;
pc->capacity = 0;
pc->sz = 0;
printf("销毁成功\n");
}
void SaveContact(const Contact* pc)
{
FILE* pf =fopen("Contact.txt", "wb");//二进制方式写进去
if (pf == NULL)
{
perror("SaveContact::fopen");
return ;
}
//写文件
int i = 0;
for (i = 0; i < pc->sz; i++)
{
fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);
}
printf("保存成功!\n");
//关闭文件
fclose(pf);
pf = NULL;
}