1.实现一个通讯录;
通讯录可以用来存储1000个人的信息,每个人的信息包括:
姓名、性别、年龄、电话、住址
提供方法:
1. 添加联系人信息
2. 删除指定联系人信息
3. 查找指定联系人信息
4. 修改指定联系人信息
5. 显示所有联系人信息
6. 清空所有联系人
7. 以名字排序所有联系人
在之前动态版本的基础上做了以下改变:因为是文件版本的,所以这里用宏定义了一个文件名,方便以后修改文件名。以及增加了两个函数:一个是保存联系人函数SaveContact(),和加载联系人函数LoadContact()。
#ifndef __CONTACT_H_
#define __CONTACT_H_
#define _CRT_SECURE_NO_WARNINGS
#define MAX_NAME 20
#define MAX_SEX 10
#define MAX_ADDR 30
#define MAX_TEL 15
#define MAX 1000//定义最大联系人的数量
#define DEFAULT_SZ 3
#define FILENAME "text.txt";
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
#include<errno.h>
enum option//通讯录的各种功能,严格按照菜单的顺序写,否则会出错
{
EXIT,
ADD,
DEL,
SEARCH,
MODIFY,
SHOW,
SORT,
EMPTY,
};
typedef struct Peoinfo//定义人的信息
{
char name[MAX_NAME];
short age;
char sex[MAX_SEX];
char addr[MAX_ADDR];
int tel[MAX_TEL];
}Peoinfo, *pPeoinfo;//pPeoinfo是指向Peoinfo结构体的一个指针
typedef struct contact
{
pPeoinfo data;//存放数据的位置
int sz;//有效元素的个数
int capacity;//容量
}contact, *pcontact;
void SaveContact(pcontact pc);//保存信息
void LoadContact(pcontact pc);//加载信息
void InitContact(pcontact pc); //初始化信息
void DistroyContact(pcontact pc);//销毁通讯录
void AddContact(pcontact pc);//添加联系人
void ShowContact(pcontact pc);//显示联系人
void DelContact(pcontact pc);//删除联系人
void ModifyContact(pcontact pc);//修改联系人
void EmptyContact(pcontact pc);//置空联系人
void SortContact(pcontact pc);//排序联系人
int SearchContact(pcontact pc);//查找联系人
void CheckCapacity(pcontact pc);//判断是否增容
#endif
contact.c
这一部分是各个函数的具体实现,在之前的基础上做了一些改变。因为是文件的版本,当打开通讯录时,当选择显示联系人时应该要显示出之前保存的联系人,所以这里需要一个加载联系人的函数LoadContact()。同时在销毁通讯录时,应该要保存之前录入的联系人,所以这里需要一个保存联系人的函数SaveContact()。而且在写入信息用了fwrite()和在读信息时用了fread()函数。我的博客里有对他们的解释可以看看。
#include"contact.h"
void LoadContact(pcontact pc)//加载联系人
{
Peoinfo temp = {0};
FILE* pfread = fopen("FILENAME", "r");//以只读的形式打开文件
if (pfread == NULL)
{
perror("open file for read");
exit(EXIT_FAILURE);//若打开文件失败,直接退出。
}
//加载联系人的信息
while (fread(&temp, sizeof(Peoinfo), 1, pfread))//读信息
{
CheckCapacity(pc);//检查是否需要扩充容量
pc->data[pc->sz] = temp;//把temp里读到的信息保存到pc->data[pc->sz]里
pc->sz++;
}
fclose(pfread);//关闭流
pfread = NULL;
}
void InitContact(pcontact pc)//初始化
{
pc->sz = 0;
pc->capacity = DEFAULT_SZ;//给定默认容量
pc->data = (pPeoinfo)malloc(pc->capacity * sizeof(Peoinfo));//开辟默认容量的字节大小
if (pc->capacity == pc->sz)
{
printf("%s\n", strerror(errno));
exit(EXIT_FAILURE);//若果增容失败就退出程序
}
LoadContact(pc);//加载已有信息
}
void SaveContact(pcontact pc)//保存联系人
{
FILE* pfWrite = fopen("FILENAME", "w");//以读写的现实所打开文件
int i = 0;
if (pfWrite == NULL)
{
perror("open file for write");
exit(EXIT_FAILURE);
}
for (i = 0; i < pc->sz; i++)
{
fwrite(pc->data + i,sizeof(Peoinfo), 1, pfWrite);//写入文件
}
fclose(pfWrite);
pfWrite = NULL;
}
void DistroyContact(pcontact pc)
{
//在销毁之前保存联系人
SaveContact(pc);
//销毁
free(pc->data);
pc->data = NULL;
pc->data = 0;
pc->sz = 0;
}
void CheckCapacity(pcontact pc)
{
if (pc->capacity == pc->sz)//当容量等于联系人个数时,扩容。
{
pPeoinfo ptr = realloc(pc->data, (pc->capacity += 2)*sizeof(Peoinfo));
if (ptr != NULL)
{
pc->data = ptr;//pc->data指向新的分配的内存
printf("增容成功");
}
}
}
void AddContact(pcontact pc)//添加联系人
{
//先检查容量,看是否需要增容
CheckCapacity(pc);
printf("请输入姓名>");
scanf("%s", pc->data[pc->sz].name);
//pc->data为数组名,pc->sz为数组的下标。通过.访问数组成员
printf("请输入电话>");
scanf("%s", pc->data[pc->sz].tel);
printf("请输入年龄>");
scanf("%d", &pc->data[pc->sz].age);
printf("请输入性别>");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入地址>");
scanf("%s", pc->data[pc->sz].addr);
(pc->sz)++;
printf("添加成功\n");
}
void ShowContact(pcontact pc)//显示所有的联系人
{
int i = 0;
printf("%-10s\t%-10s%-5s\t%-5s\t%-10s\n", "name", "tel", "age", "sex", "addr");//打印表头
for (i = 0; i < pc->sz; i++)
{
printf("%-10s\t%-10s%-5d\t%-5s\t%-10s\n",
pc->data[i].name,
pc->data[i].tel,
pc->data[i].age,
pc->data[i].sex,
pc->data[i].addr
);
}
}
static int FindEntry(pcontact pc, char name[])//查找是否存在该联系人,在删除,修改联系人中会用到
{
int i = 0;
for (i = 0; i < pc->sz; i++)
{
if (strcmp(name, pc->data[i].name) == 0)//比较联系人是否存在
{
return i;
}
}
return-1;
}
void DelContact(pcontact pc)//删除联系人
{
char name[MAX_NAME] = { 0 };
int pos = 0;
if (pc->sz == 0)
{
printf("没有联系人可以删除\n");
return;
}
printf("请输入要删除的联系人");
scanf("%s", name);
pos = FindEntry(pc, name);//查找是否存在该联系人
if (pos == -1)
{
printf("没有找到该联系人\n");
return;
}
else
{
int i = 0;
for (i = pos; i < pc->sz - 1; i++)
{
pc->data[i] = pc->data[i + 1];//让后面的联系人覆盖该联系人,达到删除的作用
}
(pc->sz)--;
printf("删除成功\n");
}
}
void EmptyContact(pcontact pc)//置空联系人
{
pc->sz = 0;
printf("清空联系人成功\n");
}
int SearchContact(pcontact pc)//查找联系人
{
int pos = 0;
char name[20];
printf("请输入要查找的联系人\n");
scanf("%s", &name);
pos = FindEntry(pc, name);//查找该联系人是否存在
if (pos == -1)
{
printf("不存在该联系人\n");
}
else
{
printf("姓名:%s\n", pc->data[pos].name);
printf("电话:%s\n", pc->data[pos].tel);
printf("年龄:%d\n", pc->data[pos].age);
printf("性别:%s\n", pc->data[pos].sex);
printf("地址:%s\n", pc->data[pos].addr);
}
return pos;
}
void menu1()
{
printf("*** 1.修改姓名 ***\n");
printf("*** 2.修改电话 ***\n");
printf("*** 3.修改年龄 ***\n");
printf("*** 4.修改性别 ***\n");
printf("*** 5.修改地址 ***\n");
printf("*** 0.退出修改 ***\n");
}
void ModifyContact(pcontact pc)//修改联系人
{
char name[20] = { 0 };
int pos = 0;
printf("请输入要修改的联系人>");
scanf("%s", &name);
pos = FindEntry(pc, name);//先查找该联系人是否存在
if (pos != -1)
{
printf("姓名:%s\n", pc->data[pos].name);
printf("电话:%s\n", pc->data[pos].tel);
printf("年龄:%d\n", pc->data[pos].age);
printf("性别:%s\n", pc->data[pos].sex);
printf("地址:%s\n", pc->data[pos].addr);
int i = 0;
do
{
menu1();
printf("请输入修改项>");
scanf("%d", &i);
switch (i)
{
case 1:
printf("把当前姓名修改为:\n");
scanf("%s", pc->data[pos].name);
break;
case 2:
printf("把当前的电话修改为:\n");
scanf("%s", pc->data[pos].tel);
break;
case 3:
printf("把当前的年龄修改为:\n");
scanf("%d", &pc->data[pos].age);
break;
case 4:
printf("把当前性别修改为:\n");
scanf("%s", pc->data[pos].sex);
break;
case 5:
printf("把当前的地址修改为:\n");
scanf("%s", pc->data[pos].addr);
break;
case 0:
break;
default:
printf("输入选项错误\n");
break;
}
} while (i);
}
else
{
printf("不存在该联系人\n");
}
}
void SortContact(pcontact pc)//对联系人按年龄排序
{
int i = 0;
int j = 0;
for (i = 0; i < pc->sz - 1; i++)
{
for (j = 0; j < pc->sz - 1 - i; j++)
{
if (pc->data[j].age< pc->data[j + 1].age)
{
Peoinfo temp;//创建一个与pc->data的类型相同存放pc->data
temp = pc->data[j];//交换两个联系人的顺序
pc->data[j] = pc->data[j + 1];
pc->data[j + 1] = temp;
}
}
}
printf("联系人排序成功\n");
}
text.c
测试文件,同之前的版本一样,由菜单函数和主函数构成。我用了switch语句去执行不同的功能。
#include"contact.h"
void menu()
{
printf("*** 1.ADD 2.DEL ***\n");
printf("*** 3.SEARCH 4.MODIFY ***\n");
printf("*** 5.SHOW 6.SORT ***\n");
printf("*** 7.EMPTY 0.EXIT ***\n");
}
int main()
{
int input;
contact my_con;
InitContact(&my_con);
//传地址的原因:
//1.结构体传参要传地址 2.结构体内部要改变函数外边的值要传地址
do
{
menu();
printf("请选择> ");
scanf("%d", &input);
switch (input)
{
case ADD:
AddContact(&my_con);
break;
case DEL:
DelContact(&my_con);
break;
case SEARCH:
SearchContact(&my_con);
break;
case MODIFY:
ModifyContact(&my_con);
break;
case SHOW:
ShowContact(&my_con);
//可以取地址,也可以不取地址。因为只是查看不是改变内容。
break;
case SORT:
SortContact(&my_con);
break;
case EMPTY:
EmptyContact(&my_con);
break;
case EXIT:
DistroyContact(&my_con);
break;
default:
printf("输入有误");
}
} while (input);
system("pause");
return 0;
}
到这三个版本的通讯录就完成了。
通讯录的静态版连接:https://blog.youkuaiyun.com/qq_40955824/article/details/81411958
通讯录的动态版连接:https://blog.youkuaiyun.com/qq_40955824/article/details/81462109