个人主页(找往期文章包括但不限于本期文章中不懂的知识点): 我要学编程(ಥ_ಥ)-CSDN博客
目录
静态通讯录的实现逻辑
test.c:通讯录的逻辑实现
Contact.h:函数的声明与头文件的包含
Contact.c:函数的实现
通讯录源码:
test.c:
Contact.c:
Contect.h:
动态版通讯录
test.c:
Contact.h:
Contact.c:
动态通讯录(文件版)
test.c:
Contact.h:
Contact.c:
静态通讯录的实现逻辑
test.c:通讯录的逻辑实现
我们今天就一起来用c语言写一个通讯录的小程序。
这个通讯录可以实现存储100个联系人。
首先,我们进去这个通讯录,肯定得有一些基本的功能。如下图:
这个就和我们在前面写游戏的时候是一样的。
void menu()
{printf("******************************************\n");printf("**** 1.增加联系人 2.删除联系人 ****\n");printf("**** 3.查找联系人 4.修改联系人 ****\n");printf("**** 5.显示联系人 6.排序联系人 ****\n");printf("**** 0.退出通讯录 ****\n");printf("******************************************\n");
}
打印出界面之后,我们就得根据需求来进行选择。
void test()
{int input = 0;//下面是对通讯录进行的各种操作,所以我们得先有一个通讯录//创建一个通讯录,包含一个人的各种信息,是一个复杂对象,用结构体来描述Contact con;InitContact(&con);//初始化通讯录do{menu();//打印基本功能printf("请选择:");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:printf("退出通讯录\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);}
为了让别人能够更清楚我们代码的功能,我们最好能把那些case后面的数字,改成ADD这些能够让别人一下就能够看懂。我们就可以联想到联合体。自定义类型:联合和枚举-CSDN博客
上面这篇博客,有关于联合体的知识。
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:SortContact(&con);//按名字排序联系人break;case EXIT:printf("退出通讯录\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);}
可以把test函数在main函数里调用。
#include "Contact.h"
int main()
{test();return 0;
}
写完这些基本的功能之后,就可以开始实现这些编辑通讯录的函数了。上面这些都是放在test.c函数(实现通讯录的基本逻辑)。
Contact.h:函数的声明与头文件的包含
这些函数的声明和头文件的声明,我们都可以放在一个Contact.h的头文件中。
头文件的声明:
//头文件的声明
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>
联合体的创建:
//把选项一 一列举出来
enum OPPION
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};
通讯录准备:
//为后期增加联系人做准备
#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30//类型的声明
typedef struct people
{char name[MAX_NAME];//姓名int age;//年龄char sex[MAX_SEX];//性别//电话(我们正常的电话是11位,超过了int的范围,所以用字符串比较合适)char tele[MAX_TELE];char addr[MAX_ADDR];//住址
}people;//通讯录的创建
typedef struct Contact
{people data[MAX];int sz;//记录当前通讯录的人数
}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 SortContact(const Contact* pc);
Contact.c:函数的实现
初始化通讯录函数:
#include "Contact.h"//声明头文件//初始化通讯录的函数
void InitContact(Contact* pc)
{assert(pc);//把pc->data的空间全部初始化为0memset(pc->data, 0, sizeof(pc->data));//循环也可以,但是太麻烦。pc->sz = 0;
}
增加联系人信息的函数:
//增加联系人信息的函数
void AddContact(Contact* pc)
{assert(pc);//首先得判断这个通讯录是否已经满了if (pc->sz == MAX){printf("通讯录已满,无法增加\n");//通讯录满了,就无需执行下面的语句了return;//因为是void。所以无需返回任何值}else{printf("请输入姓名:");scanf("%s", pc->data[pc->sz].name);printf("请输入年龄:");scanf("%d", &(pc->data[pc->sz].age));//age是int类型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++;printf("成功增加联系人\n");}
}
显示联系人的信息的函数:
//显示联系人的信息的函数
void ShowContact(const Contact* pc)
{assert(pc);//要有一定的排版(采用左对齐)printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");for (int i = 0; i < pc->sz; i++){printf("%-20s\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);}
}
删除联系人信息的函数的声明:
//删除联系人信息的函数的声明
void DelContact(Contact* pc)
{assert(pc);//判断是否能删除if (pc->sz == 0){printf("通讯录为空,无法删除\n");return;}//创建一个name数组来存放我们要寻找的联系人char name[MAX_NAME] = { 0 };//知道要删除谁printf("请输入要删除的人的名字:");scanf("%s", name);//找到要删除的人(字符串)int del = 0;int flag = 0;//遍历数组,一个一个的比较字符串for (int i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name, name) == 0){del = i;//记录当前的位置flag = 1;break;}}//开始删除(删除的方法就是把后面一个往前覆盖)if (flag == 0){printf("要删除的联系人不存在\n");return;//后面的代码无需执行}else{for (int i = del; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}}//删除最后一个元素时,虽然不能在循环中删除,但是pc->sz--了,致使访问不到最后一个元素了//因此,我们在打印时也不会打印出来pc->sz--;printf("删除成功\n");
}
其实我们会发现在这里寻找名字的时候,在后面查找联系人也会用到,不如我们直接分装成一个寻找名字的函数。
//查找联系人名字的函数
//这个函数我们只想在内部使用,没必要暴露给别人
static int FindByName(const Contact* pc, char name[])
{assert(pc);for (int 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;}char name[MAX_NAME] = { 0 };//知道要删除谁printf("请输入要删除的人的名字:");scanf("%s", name);//找到要删除的人(字符串)int del = FindByName(pc, name);if (del == -1){printf("要删除的人不存在\n");return;}for (int i = del; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}//删除最后一个元素时,虽然不能在循环中删除,但是pc->sz--了,致使访问不到最后一个元素了//因此,我们在打印时也不会打印出来pc->sz--;printf("删除成功\n");
}
查找指定联系人信息的函数:
//查找指定联系人信息的函数
void SearchContact(const Contact* pc)
{assert(pc);//存放名字的数组char name[MAX_NAME];printf("请输入要查找的联系人:");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要查找的人不存在\n");return;}printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);
}
修改指定联系人的信息:
void menu1()
{printf("*******************************\n");printf("**** 1.姓名 2.年龄 ****\n");printf("**** 3.性别 4.电话 ****\n");printf("**** 5.住址 ****\n");printf("*******************************\n");
}
//修改指定联系人的信息
void ModifyContact(Contact* pc)
{assert(pc);//存放名字的数组char name[MAX_NAME];printf("请输入要修改的联系人:");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要修改的联系人不存在\n");return;}menu1();printf("请选择要修改的选项:");int n = 0;scanf("%d", &n);switch (n){case 1:printf("请输入姓名:");scanf("%s", pc->data[pos].name);break;case 2:printf("请输入年龄:");scanf("%d", &(pc->data[pos].age));break;case 3:printf("请输入性别:");scanf("%s", pc->data[pos].sex);break;case 4:printf("请输入电话:");scanf("%s", pc->data[pos].tele);break;case 5:printf("请输入住址:");scanf("%s", pc->data[pos].addr);break;}printf("修改成功\n");//看看是否修改成功printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);
}
int cmp_peo_by_name(const void* e1, const void* e2)
{return strcmp(((people*)e1)->name, ((people*)e2)->name);
}int cmp_peo_by_age(const void* e1, const void* e2)
{return ((people*)e1)->age - ((people*)e2)->age;
}
//排序联系人
void SortContact(const Contact* pc)
{printf("请选择想要排的序:\n");printf("********************************\n");printf("***** 1.按照名字排序 ******\n");printf("***** 2.按照年龄排序 ******\n");printf("********************************\n");int n = 0;printf("请选择:");scanf("%d", &n);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_peo_by_name);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_peo_by_age);//看看是否排序成功for (int i = 0; i < pc->sz; i++){printf("%-20s\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");
}
通讯录源码:
test.c:
#include "Contact.h"void menu()
{printf("******************************************\n");printf("**** 1.增加联系人 2.删除联系人 ****\n");printf("**** 3.查找联系人 4.修改联系人 ****\n");printf("**** 5.显示联系人 6.排序联系人 ****\n");printf("**** 0.退出通讯录 ****\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:SortContact(&con);//按名字排序联系人break;case EXIT:printf("退出通讯录\n");break;default:printf("选择错误,请重新选择\n");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 AddContact(Contact* pc)
{assert(pc);if (pc->sz == MAX){printf("通讯录已满,无法增加\n");//通讯录满了,就无需执行下面的语句了return;//因为是void。所以无需返回任何值}else{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++;printf("成功增加联系人\n");}
}//显示联系人的信息的函数
void ShowContact(const Contact* pc)
{assert(pc);printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");for (int i = 0; i < pc->sz; i++){printf("%-20s\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);}
}//查找联系人名字的函数
static int FindByName(const Contact* pc, char name[])
{assert(pc);for (int 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;}char name[MAX_NAME] = { 0 };//知道要删除谁printf("请输入要删除的人的名字:");scanf("%s", name);//找到要删除的人(字符串)int del = FindByName(pc, name);if (del == -1){printf("要删除的人不存在\n");return;}for (int i = del; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}//删除最后一个元素时,虽然不能在循环中删除,但是pc->sz--了,致使访问不到最后一个元素了//因此,我们在打印时也不会打印出来pc->sz--;printf("删除成功\n");
}//查找指定联系人信息的函数
void SearchContact(const Contact* pc)
{assert(pc);char name[MAX_NAME];printf("请输入要查找的联系人:");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要查找的人不存在\n");return;}printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);
}void menu1()
{printf("*******************************\n");printf("**** 1.姓名 2.年龄 ****\n");printf("**** 3.性别 4.电话 ****\n");printf("**** 5.住址 ****\n");printf("*******************************\n");
}
//修改指定联系人的信息
void ModifyContact(Contact* pc)
{assert(pc);char name[MAX_NAME];printf("请输入要修改的联系人:");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要修改的联系人不存在\n");return;}menu1();printf("请选择要修改的选项:");int n = 0;scanf("%d", &n);switch (n){case 1:printf("请输入姓名:");scanf("%s", pc->data[pos].name);break;case 2:printf("请输入年龄:");scanf("%d", &(pc->data[pos].age));break;case 3:printf("请输入性别:");scanf("%s", pc->data[pos].sex);break;case 4:printf("请输入电话:");scanf("%s", pc->data[pos].tele);break;case 5:printf("请输入住址:");scanf("%s", pc->data[pos].addr);break;}printf("修改成功\n");printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);
}int cmp_peo_by_name(const void* e1, const void* e2)
{return strcmp(((people*)e1)->name, ((people*)e2)->name);
}int cmp_peo_by_age(const void* e1, const void* e2)
{return ((people*)e1)->age - ((people*)e2)->age;
}void SortContact(const Contact* pc)
{printf("请选择想要排的序:\n");printf("********************************\n");printf("***** 1.按照名字排序 ******\n");printf("***** 2.按照年龄排序 ******\n");printf("********************************\n");int n = 0;printf("请选择:");scanf("%d", &n);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_peo_by_name);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_peo_by_age);for (int i = 0; i < pc->sz; i++){printf("%-20s\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");
}
Contect.h:
//头文件的声明
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>//为后期增加联系人做准备
#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30//把选项一 一列举出来
enum OPPION
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};//类型的声明
typedef struct people
{char name[MAX_NAME];//姓名int age;//年龄char sex[MAX_SEX];//性别//电话(我们正常的电话是11位,超过了int的长度,所以用字符串比较合适)char tele[MAX_TELE];char addr[MAX_ADDR];//住址
}people;//通讯录
typedef struct Contact
{people data[MAX];int sz;
}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 SortContact(const Contact* pc);
动态版通讯录
有关动态内存开辟的知识点:动态内存管理-CSDN博客
上述是静态版的通讯录。
下面我们来实现一下动态版的通讯录。
目标:1. 可以实现储存通讯录人数不限。2. 默认可以放3个人的信息,如果不够,就每次增加2个人的信息(这样方便我们测试,也可以默认别的数,但数据太大,不好测试)。
首先,存放人信息的空间大小是不需要改变的,但是我们用的那个联系人数组要变成一个由malloc函数开辟的空间。其次,我们想要知道这个空间当前用了几个,还剩几个。记录用了几个,就可以用sz来记录,而还剩几个空间,就可以用count来记录一下。
//通讯录
typedef struct Contact
{people* data;//指向通讯录的那块空间int sz;//记录当前的联系人个数int count;//记录总共开辟了多少空间
}Contact;
既然通讯录里的数据变了,那么我们的初始化函数也得变化。
//初始化通讯录的函数
void InitContact(Contact* pc)
{assert(pc);//开辟一块空间给通讯录pc->data = (people*)malloc(3 * sizeof(people));if (pc->data == NULL)//空间开辟失败{perror("InitContact");return;}pc->sz = 0;pc->count = DEFAULT_SZ;//在头文件中定义一个宏,默认数据表示初始数据
}
容量变了,那我们增加联系人的函数也得发生变化。
static int CheckCount(Contact* pc)//判断是否需要增加空间
{if (pc->sz == pc->count){people *ptr = (people*)realloc(pc->data, (pc->count + INC_SZ) * sizeof(people));if (ptr == NULL){perror("CheckCount");return 0;}pc->data = ptr;pc->count += INC_SZ;printf("增容成功\n");return 1;}return 1;
}//增加联系人信息的函数
void AddContact(Contact* pc)
{assert(pc);if (CheckCount(&pc) == 0)//增加失败就不需要往下走了{return;}else{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++;printf("成功增加联系人\n");}
}
删除,查找,显示,修改,排序这些就不需要更改了。
但我们还得在退出通讯录的时候,把动态开辟的内存给释放掉。
void DestroyContact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->count = 0;pc->sz = 0;
}
test.c:
#include "Contact.h"void menu()
{printf("******************************************\n");printf("**** 1.增加联系人 2.删除联系人 ****\n");printf("**** 3.查找联系人 4.修改联系人 ****\n");printf("**** 5.显示联系人 6.排序联系人 ****\n");printf("**** 0.退出通讯录 ****\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:SortContact(&con);//按名字排序联系人break;case EXIT:DestroyContact(&con);//销毁通讯录printf("退出通讯录\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);}int main()
{test();return 0;
}
Contact.h:
//头文件的声明
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>//为后期增加联系人做准备
#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30#define DEFAULT_SZ 3
#define INC_SZ 2//把选项一 一列举出来
enum OPPION
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};//类型的声明
typedef struct people
{char name[MAX_NAME];//姓名int age;//年龄char sex[MAX_SEX];//性别//电话(我们正常的电话是11位,超过了int的长度,所以用字符串比较合适)char tele[MAX_TELE];char addr[MAX_ADDR];//住址
}people;//通讯录
typedef struct Contact
{people* data;int sz;int count;
}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 SortContact(const Contact* pc);//销毁通讯录的函数
void DestroyContact(Contact* pc);
Contact.c:
#include "Contact.h"//初始化通讯录的函数
void InitContact(Contact* pc)
{assert(pc);//开辟一块空间给通讯录pc->data = (people*)malloc(3 * sizeof(people));if (pc->data == NULL)//空间开辟失败{perror("InitContact");return;}pc->sz = 0;pc->count = DEFAULT_SZ;
}static int CheckCount(Contact* pc)//判断是否需要增加空间
{if (pc->sz == pc->count){people *ptr = (people*)realloc(pc->data, (pc->count + INC_SZ) * sizeof(people));if (ptr == NULL){perror("CheckCount");return 0;}pc->data = ptr;pc->count += INC_SZ;printf("增容成功\n");return 1;}return 1;
}//增加联系人信息的函数
void AddContact(Contact* pc)
{assert(pc);if (CheckCount(pc) == 0)//增加失败就不需要往下走了{return;}else{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++;printf("成功增加联系人\n");}
}//显示联系人的信息的函数
void ShowContact(const Contact* pc)
{assert(pc);printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");for (int i = 0; i < pc->sz; i++){printf("%-20s\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);}
}//查找联系人名字的函数
static int FindByName(const Contact* pc, char name[])
{assert(pc);for (int 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;}char name[MAX_NAME] = { 0 };//知道要删除谁printf("请输入要删除的人的名字:");scanf("%s", name); //找到要删除的人(字符串)int del = FindByName(pc, name);if (del == -1){printf("要删除的人不存在\n");return;}for (int i = del; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}//删除最后一个元素时,虽然不能在循环中删除,但是pc->sz--了,致使访问不到最后一个元素了//因此,我们在打印时也不会打印出来pc->sz--;printf("删除成功\n");
}//查找指定联系人信息的函数
void SearchContact(const Contact* pc)
{assert(pc);char name[MAX_NAME];printf("请输入要查找的联系人:");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要查找的人不存在\n");return;}printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);
}void menu1()
{printf("*******************************\n");printf("**** 1.姓名 2.年龄 ****\n");printf("**** 3.性别 4.电话 ****\n");printf("**** 5.住址 ****\n");printf("*******************************\n");
}
//修改指定联系人的信息
void ModifyContact(Contact* pc)
{assert(pc);char name[MAX_NAME];printf("请输入要修改的联系人:");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要修改的联系人不存在\n");return;}menu1();printf("请选择要修改的选项:");int n = 0;scanf("%d", &n);switch (n){case 1:printf("请输入姓名:");scanf("%s", pc->data[pos].name);break;case 2:printf("请输入年龄:");scanf("%d", &(pc->data[pos].age));break;case 3:printf("请输入性别:");scanf("%s", pc->data[pos].sex);break;case 4:printf("请输入电话:");scanf("%s", pc->data[pos].tele);break;case 5:printf("请输入住址:");scanf("%s", pc->data[pos].addr);break;}printf("修改成功\n");printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);
}int cmp_peo_by_name(const void* e1, const void* e2)
{return strcmp(((people*)e1)->name, ((people*)e2)->name);
}int cmp_peo_by_age(const void* e1, const void* e2)
{return ((people*)e1)->age - ((people*)e2)->age;
}void SortContact(const Contact* pc)
{printf("请选择想要排的序:\n");printf("********************************\n");printf("***** 1.按照名字排序 ******\n");printf("***** 2.按照年龄排序 ******\n");printf("********************************\n");int n = 0;printf("请选择:");scanf("%d", &n);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_peo_by_name);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_peo_by_age);for (int i = 0; i < pc->sz; i++){printf("%-20s\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 DestroyContact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->count = 0;pc->sz = 0;
}
动态通讯录(文件版)
有关文件的读写的知识点:C语言之文件操作(万字详解)-CSDN博客
上面是动态版的通讯录,可以实现空间不受限制。但是如果我们想要知道我们上一次是存放什么呢?那就得用文件来保存上一次的信息。我们应该是在销毁前保存。
void SaveContact(Contact* pc)
{//打开文件FILE* pf = fopen("data.txt", "wb");if (pf == NULL){perror("SaveContact");return;}//写数据for (int i = 0; i < pc->sz; i++){fwrite(pc->data + i, sizeof(people), 1, pf);}//关闭文件fclose(pf);pf = NULL;
}
我们保存的信息是为了下一次打开的时候,可以直接查看到这些数据。那么我们在初始化通讯录的时候,就应该把数据放到通讯录里,如果我们要查看的话,就可以选择显示通讯录的信息即可。
void LoadContact(Contact* pc)
{//把文件中的信息加载到通讯录中FILE* pf = fopen("data.txt", "rb");if (pf == NULL){perror("LoadContact");return;}//读文件people tmp = {0};while (fread(&tmp, sizeof(people), 1, pf))//读一个判断一个,当返回为0时,就说明已经读完了{if (CheckCount(pc) == 0)//存放的时候,判断是否需要增加空间{return;}pc->data[pc->sz] = tmp;pc->sz++;}//关闭文件fclose(pf);pf = NULL;
}//初始化通讯录的函数
void InitContact(Contact* pc)
{assert(pc);//开辟一块空间给通讯录pc->data = (people*)malloc(3 * sizeof(people));if (pc->data == NULL)//空间开辟失败{perror("InitContact");return;}pc->sz = 0;pc->count = DEFAULT_SZ;//把文件中的信息加载到通讯录中LoadContact(pc);
}
test.c:
#include "Contact.h"void menu()
{printf("******************************************\n");printf("**** 1.增加联系人 2.删除联系人 ****\n");printf("**** 3.查找联系人 4.修改联系人 ****\n");printf("**** 5.显示联系人 6.排序联系人 ****\n");printf("**** 0.退出通讯录 ****\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:SortContact(&con);//按名字排序联系人break;case EXIT:SaveContact(&con);//把信息保存在文件里DestroyContact(&con);//销毁通讯录printf("退出通讯录\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);}int main()
{test();return 0;
}
Contact.h:
//头文件的声明
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>//为后期增加联系人做准备
#define MAX 100
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30#define DEFAULT_SZ 3
#define INC_SZ 2//把选项一 一列举出来
enum OPPION
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};//类型的声明
typedef struct people
{char name[MAX_NAME];//姓名int age;//年龄char sex[MAX_SEX];//性别//电话(我们正常的电话是11位,超过了int的长度,所以用字符串比较合适)char tele[MAX_TELE];char addr[MAX_ADDR];//住址
}people;//通讯录
typedef struct Contact
{people* data;int sz;int count;
}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 SortContact(const Contact* pc);//销毁通讯录的函数
void DestroyContact(Contact* pc);//加载通讯录的函数
void SaveContact(Contact* pc);
Contact.c:
#include "Contact.h"int CheckCount(Contact* pc);void LoadContact(Contact* pc)
{//把文件中的信息加载到通讯录中FILE* pf = fopen("data.txt", "rb");if (pf == NULL){perror("LoadContact");return;}//读文件people tmp = {0};while (fread(&tmp, sizeof(people), 1, pf)){if (CheckCount(pc) == 0){return;}pc->data[pc->sz] = tmp;pc->sz++;}//关闭文件fclose(pf);pf = NULL;
}//初始化通讯录的函数
void InitContact(Contact* pc)
{assert(pc);//开辟一块空间给通讯录pc->data = (people*)malloc(3 * sizeof(people));if (pc->data == NULL)//空间开辟失败{perror("InitContact");return;}pc->sz = 0;pc->count = DEFAULT_SZ;//把文件中的信息加载到通讯录中LoadContact(pc);
}static int CheckCount(Contact* pc)//判断是否需要增加空间
{if (pc->sz == pc->count){people *ptr = (people*)realloc(pc->data, (pc->count + INC_SZ) * sizeof(people));if (ptr == NULL){perror("CheckCount");return 0;}pc->data = ptr;pc->count += INC_SZ;printf("增容成功\n");return 1;}return 1;
}//增加联系人信息的函数
void AddContact(Contact* pc)
{assert(pc);if (CheckCount(pc) == 0)//增加失败就不需要往下走了{return;}else{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++;printf("成功增加联系人\n");}
}//显示联系人的信息的函数
void ShowContact(const Contact* pc)
{assert(pc);printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");for (int i = 0; i < pc->sz; i++){printf("%-20s\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);}
}//查找联系人名字的函数
static int FindByName(const Contact* pc, char name[])
{assert(pc);for (int 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;}char name[MAX_NAME] = { 0 };//知道要删除谁printf("请输入要删除的人的名字:");scanf("%s", name);//找到要删除的人(字符串)int del = FindByName(pc, name);if (del == -1){printf("要删除的人不存在\n");return;}for (int i = del; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}//删除最后一个元素时,虽然不能在循环中删除,但是pc->sz--了,致使访问不到最后一个元素了//因此,我们在打印时也不会打印出来pc->sz--;printf("删除成功\n");
}//查找指定联系人信息的函数
void SearchContact(const Contact* pc)
{assert(pc);char name[MAX_NAME];printf("请输入要查找的联系人:");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要查找的人不存在\n");return;}printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);
}void menu1()
{printf("*******************************\n");printf("**** 1.姓名 2.年龄 ****\n");printf("**** 3.性别 4.电话 ****\n");printf("**** 5.住址 ****\n");printf("*******************************\n");
}
//修改指定联系人的信息
void ModifyContact(Contact* pc)
{assert(pc);char name[MAX_NAME];printf("请输入要修改的联系人:");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要修改的联系人不存在\n");return;}menu1();printf("请选择要修改的选项:");int n = 0;scanf("%d", &n);switch (n){case 1:printf("请输入姓名:");scanf("%s", pc->data[pos].name);break;case 2:printf("请输入年龄:");scanf("%d", &(pc->data[pos].age));break;case 3:printf("请输入性别:");scanf("%s", pc->data[pos].sex);break;case 4:printf("请输入电话:");scanf("%s", pc->data[pos].tele);break;case 5:printf("请输入住址:");scanf("%s", pc->data[pos].addr);break;}printf("修改成功\n");printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);
}int cmp_peo_by_name(const void* e1, const void* e2)
{return strcmp(((people*)e1)->name, ((people*)e2)->name);
}int cmp_peo_by_age(const void* e1, const void* e2)
{return ((people*)e1)->age - ((people*)e2)->age;
}void SortContact(const Contact* pc)
{printf("请选择想要排的序:\n");printf("********************************\n");printf("***** 1.按照名字排序 ******\n");printf("***** 2.按照年龄排序 ******\n");printf("********************************\n");int n = 0;printf("请选择:");scanf("%d", &n);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_peo_by_name);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_peo_by_age);for (int i = 0; i < pc->sz; i++){printf("%-20s\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 DestroyContact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->count = 0;pc->sz = 0;
}void SaveContact(Contact* pc)
{//打开文件FILE* pf = fopen("data.txt", "wb");if (pf == NULL){perror("SaveContact");return;}//写数据for (int i = 0; i < pc->sz; i++){fwrite(pc->data + i, sizeof(people), 1, pf);}//关闭文件fclose(pf);pf = NULL;
}
好啦!本期的通讯录的实现就到此结束啦!下一期,我们再一起学习吧!