C语言——通讯录
前言:
要实现通讯录的增删查改,动态管理,文件操作,你首先需要掌握以下内容:
头文件定义
所需头文件:
#include<stdio.h> //用于标准输入输出 #include<stdlib.h> //用于动态内存开辟 #include<stdbool.h> //用于返回bool类型 #include<assert.h> //用于使用断言 #include<string.h> //用于字符串操作函数
为了方便后续的修改,我们定义几个全局变量来确定每个成员信息的长度大小
#define Max_Number_Init 3 //初始可容纳成员个数 #define Max_Name 20 //成员姓名的最大长度 #define Max_Sex 5 //成员性别的最大长度 #define Max_PhoneNumber 15 //成员电话的最大长度 #define Max_Address 12 //成员地址的最大长度
定义成员信息的结构体:
typedef struct PeopleInformation { char name[Max_Name]; //姓名 int age; //年龄 char sex[Max_Sex]; //性别 char PhoneNumber[Max_PhoneNumber]; //电话 char address[Max_Address]; //地址 }PeoInfo;
定义通讯录的结构体:
typedef struct Contact { PeoInfo* nums; //储存成员信息的数组 int size; //记录通讯录已经容纳的成员个数 int capacity; //记录通讯录的最大容量 }Con;
实现主函数(int main())
打印选择菜单
void meau() { printf("*******************************\n"); printf("****1->Add 2->Del******\n"); printf("****3->Search 4->Modify***\n"); printf("****5->Show 6->Claer****\n"); printf("****0->Exit **********\n"); }
为了提高代码的可读性,我们可以定义一个枚举类型,来分别代表这些选项:
enum Choice { Exit, //0 Add, //1 Del, //2 Search, //3 Modify, //4 Show, //5 Clear //6 };
int main()
int main() { int input = 0; //为通讯录申请内存 Con* con = (Con*)malloc(sizeof(Con)); ContactInit(con); //初始化通讯录 do { meau(); printf("请输入需要的选项:"); scanf("%d", &input); switch (input) { case Add: ContactPush(con); //增 break; case Del: ContactPop(con); //删 break; case Search: ContactSearch(con); //查 break; case Modify: ContactModify(con); //改 break; case Show: ContactShow(con); //展示通讯录 break; case Clear: ContactClear(con); //清空通讯录 break; case Exit: ContactDataInput(con); //退出通讯录 printf("退出通讯录\n"); break; default: printf("输入错误,请重新输入\n"); break; } } while (input); //释放动态内存 if(con->nums) free(con->nums); free(con); return 0; }
ContactClear()
void ContactClear(Con* con)
清空通讯录,就是将保存成员信息的数组con->nums
释放,同时置空,并且将size
置0
void ContactClear(Con* con) { free(con->nums); con->nums = NULL; con->size = 0; }
ContactInit()
void ContactInit(Con* con);
通讯录的初始化就是为储存成员信息的数组申请内存,确定最大容量以及将size
置0
void ContactInit(Con* con) { con->capacity = Max_Number_Init; con->nums = (PeoInfo*)malloc(sizeof(PeoInfo) * con->capacity); con->size = 0; }
ContactPush()
void ContactPush(Con* con)
在加入成员之前,我们需要检查是否还有空间可以储存,如果已经满了,就要增容
我们可以单独写一个函数来进行检查和增容:
ContactCheak()
bool ContactFull(const Con* con) { assert(con); return con->size == con->capacity; } void ContactCheak(Con* con) { assert(con); if (ContactFull(con)) { con->capacity += 2; //如果满了,每次就将最大容量加2 //增容 PeoInfo* temp = (PeoInfo*)realloc(con->nums, con->capacity * sizeof(PeoInfo)); if (temp == NULL) { perror("realloc"); exit(0); } con->nums = temp; } }
ContactPush()
void ContactPush(Con* con) { assert(con); ContactCheak(con); //如果通讯录之前被清空过,要想重新使用,就先要为nums申请内存 if (NULL == con->nums) { con->nums = (PeoInfo*)malloc(sizeof(PeoInfo) * Max_Number_Init); if (NULL == con->nums) { perror("malloc"); exit(1); } } printf("请输入添加的联系人姓名:"); scanf("%s", (con->nums + con->size)->name); printf("请输入该联系人的年龄:"); scanf("%d", &(con->nums + con->size)->age); printf("请输入该联系人的性别:"); scanf("%s", (con->nums + con->size)->sex); printf("请输入该联系人的电话:"); scanf("%s", (con->nums + con->size)->PhoneNumber); printf("请输入该联系人的地址:"); scanf("%s", (con->nums + con->size)->address); con->size++; printf("成功添加联系人\n"); }
ContactPop()
void ContactPop(Con* con)
这里通过姓名来删除联系人
- 最开始,要判断通讯录是否为空,如果为空,就不能删除
- 在删除之前,要现在通讯录中查找是否有这个联系人,如果没有,就不要删除(我们用一个函数
ContactFind
来实现查找成员,并返回成员下标) - 为了保证删除效率,我们找到需要删除的联系人的下标后,将其和最后一个成员交换位置,再利用
size--
删除最后一个成员即可
ContactEmpty()
bool ContactEmpty(const Con* con) { assert(con); return con->size == 0; }
Swap()
void Swap(PeoInfo* peo1, PeoInfo* peo2) { PeoInfo temp = *peo1; *peo1 = *peo2; *peo2 = temp; }
ContactFind()
int ContactFind(const Con* con, const char* str) { for (int i = 0; i < con->size; i++) { if (strcmp(con->nums[i].name, str) == 0) return i; } return -1; }
ContactPop()
void ContactPop(Con* con) { assert(con); //判断指针有效性 assert(!ContactEmpty(con)); //通讯录不为空 char name[20]; printf("请输入被查找人的姓名:"); scanf("%s", name); int index = ContactFind(con, name); if (index != -1) { //打印将被删除的成员信息 printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址"); printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n", (con->nums + index)->name, (con->nums + index)->age, (con->nums + index)->sex, (con->nums + index)->PhoneNumber, (con->nums + index)->address); //清空缓冲区 while (getchar() != '\n'); printf("确定删除(Y|N): "); char ch = getchar(); if (ch == 'Y') { Swap(&con->nums[index], &con->nums[con->size - 1]); con->size--; printf("删除成功\n"); } else { printf("删除失败\n"); } } else { printf("未找到该联系人\n"); } }
ContactSearch()
void ContactSearch(const Con* con)
这里统一用姓名来查找联系人,找到便打印联系人信息
void ContactSearch(const Con* con) { assert(con); char name[20]; printf("请输入被查找人的姓名:"); scanf("%s", name); int index = ContactFind(con, name); if (index != -1) { printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址"); printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n", (con->nums + index)->name, (con->nums + index)->age, (con->nums + index)->sex, (con->nums + index)->PhoneNumber, (con->nums + index)->address); } else { printf("未找到该联系人\n"); } }
ContactModify()
void ContactModify(Con* con)
这里统一用姓名查找联系人,然后再进行修改
void ContactModify(Con* con) { assert(con); char name[20]; printf("请输入被查找人的姓名:"); scanf("%s", name); int index = ContactFind(con, name); if (index != -1) { char ch; //展示联系人姓名 printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址"); printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n\n", (con->nums + index)->name, (con->nums + index)->age, (con->nums + index)->sex, (con->nums + index)->PhoneNumber, (con->nums + index)->address); printf("是否修改年龄(Y|N): "); while (getchar() != '\n'); ch = getchar(); if (ch == 'Y') { printf("请输入新的联系人年龄:"); scanf("%d", &con->nums[index].age); } printf("是否修改性别(Y|N): "); while (getchar() != '\n'); ch = getchar(); if (ch == 'Y') { printf("请输入新的联系人性别:"); scanf("%s", con->nums[index].sex); } printf("是否修改电话(Y|N): "); while (getchar() != '\n'); ch = getchar(); if (ch == 'Y') { printf("请输入新的联系人电话:"); scanf("%s", con->nums[index].PhoneNumber); } printf("是否修改住址(Y|N): "); while (getchar() != '\n'); ch = getchar(); if (ch == 'Y') { printf("请输入新的联系人地址:"); scanf("%s", con->nums[index].address); } } else { printf("未找到该联系人\n"); } }
ContactShow()
- 展示联系人信息
- 这里我们再加一个功能,即展示的时候先将所有联系人按姓名排序,在进行打印
ContactSort()
int compare_by_name(const void* data1, const void* data2) { return strcmp(((PeoInfo*)data1)->name, ((PeoInfo*)data2)->name); } void ContactSort(Con* con) { //用qsort进行排序 qsort(con->nums, con->size, sizeof(PeoInfo), compare_by_name); }
ContactShow()
void ContactShow(const Con* con) { assert(con); ContactSort(con); printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n","姓名","年龄","性别","电话","地址"); for (int i = 0; i < con->size; i++) { printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n", (con->nums + i)->name, (con->nums + i)->age, (con->nums + i)->sex, (con->nums + i)->PhoneNumber, (con->nums + i)->address); } }
文件操作
要实现文件操作,我们再加入两个函数就可以了
- 一个是打算退出通讯录的时候,我们将所有联系人的信息都存入文件(
ContactDataInput()
) - 一个是打开通讯录,并进行初始化的时候,我们先将文件中的联系人信息存入通讯录(
ContactDataCopy()
)
ContactDataInput()
void ContactDataInput(Con* con) { assert(con); FILE* fp = fopen("data.txt", "w"); if (NULL == fp) { perror("fopen"); exit(1); } for (int i = 0; i < con->size; i++) fprintf(fp, "%s %d %s %s %s %c", con->nums[i].name, con->nums[i].age, con->nums[i].sex, con->nums[i].PhoneNumber, con->nums[i].address, '\n'); fclose(fp); fp = NULL; }
ContactDataCopy()
void ContactDataCopy(Con* con) { assert(con); FILE* fp = fopen("data.txt", "r"); if (NULL == fp) { perror("fopen"); exit(1); } PeoInfo temp = {0}; while (fscanf(fp, "%s %d %s %s %s", temp.name, &temp.age, temp.sex, temp.PhoneNumber, temp.address)!=EOF) { //插入之前先检查容量 ContactCheak(con); con->nums[con->size] = temp; con->size++; } fclose(fp); fp = NULL; }
实现代码:
#pragma once #include<stdio.h> #include<stdlib.h> #include<stdbool.h> #include<assert.h> #include<string.h> #define Max_Number_Init 3 #define Max_Name 20 #define Max_Sex 5 #define Max_PhoneNumber 15 #define Max_Address 12 typedef struct PeopleInformation { char name[Max_Name]; int age; char sex[Max_Sex]; char PhoneNumber[Max_PhoneNumber]; char address[Max_Address]; }PeoInfo; typedef struct Contact { PeoInfo* nums; int size; int capacity; }Con; void ContactInit(Con* con); bool ContactEmpty(const Con* con); bool ContactFull(const Con* con); void ContactPush(Con* con); void ContactPop(Con* con); void ContactSearch(const Con* con); void ContactModify(Con* con); void ContactShow(const Con* con); void ContactClear(Con* con); void ContactDataInput(Con* con); void ContactDataCopy(Con* con); void ContactCheak(Con* con) { assert(con); if (ContactFull(con)) { con->capacity += 2; PeoInfo* temp = (PeoInfo*)realloc(con->nums, con->capacity * sizeof(PeoInfo)); if (temp == NULL) { perror("realloc"); exit(0); } con->nums = temp; } } void ContactDataCopy(Con* con) { assert(con); FILE* fp = fopen("data.txt", "r"); if (NULL == fp) { perror("fopen"); exit(1); } PeoInfo temp = {0}; while (fscanf(fp, "%s %d %s %s %s", temp.name, &temp.age, temp.sex, temp.PhoneNumber, temp.address)!=EOF) { ContactCheak(con); con->nums[con->size] = temp; con->size++; } fclose(fp); fp = NULL; } int compare_by_name(const void* data1, const void* data2) { return strcmp(((PeoInfo*)data1)->name, ((PeoInfo*)data2)->name); } void ContactSort(Con* con) { qsort(con->nums, con->size, sizeof(PeoInfo), compare_by_name); } int ContactFind(const Con* con, const char* str) { for (int i = 0; i < con->size; i++) { if (strcmp(con->nums[i].name, str) == 0) return i; } return -1; } void ContactInit(Con* con) { con->capacity = Max_Number_Init; con->nums = (PeoInfo*)malloc(sizeof(PeoInfo) * con->capacity); con->size = 0; ContactDataCopy(con); } bool ContactEmpty(const Con* con) { assert(con); return con->size == 0; } bool ContactFull(const Con* con) { assert(con); return con->size == con->capacity; } void ContactShow(const Con* con) { assert(con); ContactSort(con); printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n","姓名","年龄","性别","电话","地址"); for (int i = 0; i < con->size; i++) { printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n", (con->nums + i)->name, (con->nums + i)->age, (con->nums + i)->sex, (con->nums + i)->PhoneNumber, (con->nums + i)->address); } } void ContactPush(Con* con) { assert(con); ContactCheak(con); if (NULL == con->nums) { con->nums = (PeoInfo*)malloc(sizeof(PeoInfo) * Max_Number_Init); if (NULL == con->nums) { perror("malloc"); exit(1); } } printf("请输入添加的联系人姓名:"); scanf("%s", (con->nums + con->size)->name); printf("请输入该联系人的年龄:"); scanf("%d", &(con->nums + con->size)->age); printf("请输入该联系人的性别:"); scanf("%s", (con->nums + con->size)->sex); printf("请输入该联系人的电话:"); scanf("%s", (con->nums + con->size)->PhoneNumber); printf("请输入该联系人的地址:"); scanf("%s", (con->nums + con->size)->address); con->size++; printf("成功添加联系人\n"); } void Swap(PeoInfo* peo1, PeoInfo* peo2) { PeoInfo temp = *peo1; *peo1 = *peo2; *peo2 = temp; } void ContactPop(Con* con) { assert(con); assert(!ContactEmpty(con)); char name[20]; printf("请输入被查找人的姓名:"); scanf("%s", name); int index = ContactFind(con, name); if (index != -1) { printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址"); printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n", (con->nums + index)->name, (con->nums + index)->age, (con->nums + index)->sex, (con->nums + index)->PhoneNumber, (con->nums + index)->address); while (getchar() != '\n'); printf("确定删除(Y|N): "); char ch = getchar(); if (ch == 'Y') { Swap(&con->nums[index], &con->nums[con->size - 1]); con->size--; printf("删除成功\n"); } else { printf("删除失败\n"); } } else { printf("未找到该联系人\n"); } } void ContactSearch(const Con* con) { assert(con); char name[20]; printf("请输入被查找人的姓名:"); scanf("%s", name); int index = ContactFind(con, name); if (index != -1) { printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址"); printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n", (con->nums + index)->name, (con->nums + index)->age, (con->nums + index)->sex, (con->nums + index)->PhoneNumber, (con->nums + index)->address); } else { printf("未找到该联系人\n"); } } void ContactModify(Con* con) { assert(con); char name[20]; printf("请输入被查找人的姓名:"); scanf("%s", name); int index = ContactFind(con, name); if (index != -1) { char ch; printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址"); printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n\n", (con->nums + index)->name, (con->nums + index)->age, (con->nums + index)->sex, (con->nums + index)->PhoneNumber, (con->nums + index)->address); printf("是否修改年龄(Y|N): "); while (getchar() != '\n'); ch = getchar(); if (ch == 'Y') { printf("请输入新的联系人年龄:"); scanf("%d", &con->nums[index].age); } printf("是否修改性别(Y|N): "); while (getchar() != '\n'); ch = getchar(); if (ch == 'Y') { printf("请输入新的联系人性别:"); scanf("%s", con->nums[index].sex); } printf("是否修改电话(Y|N): "); while (getchar() != '\n'); ch = getchar(); if (ch == 'Y') { printf("请输入新的联系人电话:"); scanf("%s", con->nums[index].PhoneNumber); } printf("是否修改住址(Y|N): "); while (getchar() != '\n'); ch = getchar(); if (ch == 'Y') { printf("请输入新的联系人地址:"); scanf("%s", con->nums[index].address); } } else { printf("未找到该联系人\n"); } } void ContactClear(Con* con) { free(con->nums); con->nums = NULL; con->size = 0; } void ContactDataInput(Con* con) { assert(con); FILE* fp = fopen("data.txt", "w"); if (NULL == fp) { perror("fopen"); exit(1); } for (int i = 0; i < con->size; i++) fprintf(fp, "%s %d %s %s %s %c", con->nums[i].name, con->nums[i].age, con->nums[i].sex, con->nums[i].PhoneNumber, con->nums[i].address, '\n'); fclose(fp); fp = NULL; } void meau() { printf("*******************************\n"); printf("****1->Add 2->Del******\n"); printf("****3->Search 4->Modify***\n"); printf("****5->Show 6->Claer****\n"); printf("****0->Exit **********\n"); } enum Choice { Exit, Add, Del, Search, Modify, Show, Clear }; int main() { int input = 0; Con* con = (Con*)malloc(sizeof(Con)); ContactInit(con); do { meau(); printf("请输入需要的选项:"); scanf("%d", &input); switch (input) { case Add: ContactPush(con); break; case Del: ContactPop(con); break; case Search: ContactSearch(con); break; case Modify: ContactModify(con); break; case Show: ContactShow(con); break; case Clear: ContactClear(con); break; case Exit: ContactDataInput(con); printf("退出通讯录\n"); break; default: printf("输入错误,请重新输入\n"); break; } } while (input); if(con->nums) free(con->nums); free(con); return 0; }