👻4.通讯录文件contact.c:👻
接下来,我将带大家一一实现该文件的每个函数!
4.1初始化通讯录函数:
使用memeset函数将p->data所指向的空间中的数据初始化为0
操作的空间大小为sizeof函数所计算出的p->data所指向空间的大小,即整个联系人数据的结构体数组data所占的空间的大小
void Initcontact(contact* con) { assert(con); con->sz = 0; memset(con->data, 0, sizeof(con->data)); }
4.2增加联系人函数:
函数功能部分很简单,每次询问一项联系人信息,然后由用户输入该项信息,并向该联系人保存地址处存入该信息。但是在这个期间仍有很多的细节需要我们去注意。
第一个要点是信息存入的地址,我们对于通讯录的访问是通过指针进行的,所以信息的存入地址一定是对应的存入地址,数组应当使用指针进行访问,变量则应当使用取地址操作符进行访问;
第二点数组的下标恰好与我们初始化的统计变量 sz 相同,并且在每次存储完联系人信息后,我们对会将用于统计联系人数量的统计变量sz 加上一,这时它又变成了下一次需要访问时的结构体数组的下标。
第三点我们在进行存储前还应当进行一次判断,即判断通讯簿内是否已经存满了联系人信息,如果已经存满了联系人信息,则应当提示用户没有足够的空间;
void AddContact(contact* con) { assert(con); if (con->sz == 99) { printf("通讯录已经存满,请联系操作人员"); return; } printf("请输入联系人姓名:>"); scanf("%s", con->data[con->sz].name); printf("请输入联系人性别:>"); scanf("%s",con->data[con->sz].sex); printf("请输入联系人年龄:>"); scanf("%d", &(con->data[con->sz].age)); printf("请输入联系人联系方式:>"); scanf("%s", con->data[con->sz].tele); printf("请输入联系人住址:>"); scanf("%s", con->data[con->sz].addr); printf("联系人信息添加成功!\n"); con->sz++; }
4.3删除联系人函数:
在想要删除联系人信息时,我们首先应当对通讯录中的内容进行判断,如果此时通讯录中没有存储任何联系人的信息,则应当提示用户没有联系人信息(后面都会有这个判断,就不一一讲解了)
if (p->sz == 0) { printf("当前通讯录为空!\n"); }
查找联系人的逻辑很简单,我们只需要定义一个字符串,然后让用户输入想要查找的联系人的姓名,用一个函数来查找位置即可。
当删除时,我们的实现方式是,让选中联系人的后一个联系人将其覆盖,并以此类推至后面的所有联系人,并在完成后,将统计变量减一,即缩短联系人信息数组。如果是最后一个人需要被删除,则不用管他。在循环中一定要注意是否越界!这是编程的好习惯!
printf("请输入要删除人名字:>"); scanf("%s", name); int ret = FindByName(con, name); int i = 0; if (-1 == ret) { printf("没有找到此人\n"); return; } else { for (i = ret; i < con->sz-1; i++) { con->data[i] = con->data[i + 1]; } con->sz--; memset(&con->data[i], 0, sizeof(con->data[i])); printf("删除成功!\n"); } }
4.3.1查找姓名函数:
接着我们就可以将通讯录中的联系人姓名与用户输入的姓名,使用遍历与 strcmp 函数进行比较.找到返回下标,没找到返回-1:(该函数后面也都会用到,后面就不一一讲解)
int FindByName(contact* con, char name[]) { assert(con); int i = 0; for (i = 0; i < con->sz; i++) { if (strcmp(con->data[i].name,name) == 0) { return i; } } return -1; }
4.4查找联系人函数:
这个函数很简单,就是后面打印的时候要注意这些知识点:
%s,%d等在d前面加数字表示有多少位,如果这位置没有数字,就补为空格。
数字为+数字表示右对齐,加-号表示左对齐。
我们可以将字符串常量以%s形式打印出来,以此控制长度。
void SeachContact(contact* con) { assert(con); char name[MAX_NAME]; if (con->sz == 0) { printf("通讯录为空,无法输入\n"); return; } printf("请输入查找人名字"); scanf("%s", name); int ret = FindByName(con, name); if (ret == -1) { printf("您查找的人不存在!\n"); return; } printf("%-20s\t%-4s\t%-5s\t%-20s\t%-12s\n", "姓名", "年龄", "性别", "地址", "电话"); printf("%-20s\t%-4d\t%-5s\t%-20s\t%-12s\n", con->data[ret].name, con->data[ret].age, con->data[ret].sex, con->data[ret].addr, con->data[ret].tele); }
4.5修改联系人函数:
此函数就是在找到相应联系人后修改即可
void ModifyContact(contact* con) { assert(con); char name[MAX_NAME] = { 0 }; if (con->sz == 0) { printf("通讯录为空,无法修改\n"); return; } printf("请输入查找人名字"); scanf("%s", name); int ret = FindByName(con, name); if (-1 == ret) { printf("您要修改的人不存在\n"); return; } printf("请输入联系人姓名:>"); scanf("%s", con->data[ret].name); printf("请输入联系人性别:>"); scanf("%s", con->data[ret].sex); printf("请输入联系人年龄:>"); scanf("%d", &(con->data[ret].age)); printf("请输入联系人住址:>"); scanf("%s", con->data[ret].addr); printf("请输入联系人联系方式:>"); scanf("%s", con->data[ret].tele); printf("联系人信息修改成功!\n"); }
4.6打印通讯录函数:
这个函数就是通过循环打印所有的消息即可,也非常简单:
void PrintContact(contact* con) { printf("%-10s\t%-20s\t%-4s\t%-5s\t%-20s\t%-12s\n","联系人", "姓名", "年龄", "性别", "地址", "电话"); for (int i = 0; i < con->sz; i++) { printf("%-10d\t%-20s\t%-4d\t%-5s\t%-20s\t%-12s\n",i+1, con->data[i].name, con->data[i].age, con->data[i].sex, con->data[i].addr, con->data[i].tele); } }
4.7按名字排序通讯录函数:
这里我们就可以用到我们所学的qsort函数了,忘记的家人们可以看看之前写的博客:qsort函数。这里需要注意的是,在写比较大小函数中时,强转的类型是PeoInfo,因为我们是对通讯录排序,对那个结构体数组排序,并且要是升序。这些知识点相信大家复习完就可以知道。
int cmp_con_by_name(const void* e1, const void* e2) { return (strcmp(((PeoInfo *)e1)->name, ((PeoInfo*)e2)->name)); } void sortContact(contact* con) { assert(con); if (0 == con->sz) { printf("通讯录为空,无法排序\n"); return; } qsort(con->data, con->sz, sizeof(con->data[0]), cmp_con_by_name); printf("排序成功"); }