• c语言实现通讯录(用三种方法来实现一个属于你的通讯录)


    前沿:


      ~~在一个周前,我用c语言实现了一个银行账号的功能,在总结当中我说了要实现一个通讯录,来实现人员的增删插改功能,而现在就是我实现它的时候!!,本文呢小马将通过三种形式来实现用户的增删插改,其实也就是一点点的优化版本,从静态的实现,到动态的实现,最后以文件的形式来完成,请大家和我一起往下看吧!➡︎ 

     思维导图:↘️      ⤵️

    ↘️

    目录

    一、基础版本:

    1.1 通讯录的个人信息(结构体来实现):

    1..2通讯录名单:

    1.3人员初始化:

    1.4菜单:

    1.5主函数:

    二、功能的实现:

    2.1、增加人数:

    2.2、删除人数:

    2.3、查找:

    2.4、展示:

    2.5、排序(这里我是通过名字):

    三、通讯录进阶(设置动态存储):

    3.1通讯录从静态改为动态:

    3.2通讯录的初始化:

    3.3通讯录的增加需要判断是否满了:

    四、文件的形式存储通讯录:

    4.1人员信息的保存:

    4.2人员信息的流入:

    总结:


     

    一、基础版本:

    💖💖前提准备: 

    1、通讯录里面的人的个人信息(姓名、性别、年龄、手机号、地址)。

    2、通讯录名单(来实现人员的增删插改)。

    3、人员的初始化。

    4、菜单。

    1.1 通讯录的个人信息(结构体来实现):

    1. typedef struct peoInfo //重定义
    2. {
    3. char name[20]; //姓名
    4. int age ; //年龄
    5. char sex[10]; //性别
    6. char tele[12]; //手机号
    7. char addr[30]; //地址
    8. }peoInfo;

    1..2通讯录名单:

    1. typedef struct Contact
    2. {
    3. peoInfo data[100]; //人员名单 结构体数组
    4. int count; //人员数量
    5. }Contact;

    1.3人员初始化:

    1. void InitContact(Contact* con)
    2. {
    3. assert(con);
    4. con->count=0; //初始人数为0
    5. memset(con->data,0,sizeof(con->data)); //这是一个c语言的库函数把数组里面的数全部初始化为0
    6. }

    1.4菜单:

    1. void menu(void)
    2. {
    3. printf("===============================\n");
    4. printf("====1.add 2. del 3.serch=====\n");
    5. printf("=====4.modify 5.show======\n");
    6. printf("=====6.sort 0.exit======\n");
    7. printf("===============================\n");
    8. printf("===============================\n");
    9. }

    1.5主函数:

    1. enum a
    2. {
    3. ext, //这里用了简单的枚举的方法 枚举的初始值为0
    4. add,
    5. del,
    6. serch,
    7. modify,
    8. show,
    9. sort
    10. };
    11. int main()
    12. {
    13. int choice;
    14. Contact con; //通讯录
    15. do
    16. {
    17. menu();
    18. printf("请选择功能\n");
    19. scanf("%d",&choice);
    20. switch(choice)
    21. {
    22. case add:
    23. AddContact(&con); //实现增加功能的函数
    24. break ;
    25. case del:
    26. DelContanct(&con); //实现删除功能的函数
    27. break ;
    28. case serch:
    29. SerContact(&con); //实现查找功能的函数
    30. break ;
    31. case modify:
    32. ModContact(& con); //实现修改功能的函数
    33. break ;
    34. case show:
    35. ShowContact(&con); //实现展示功能的函数
    36. break ;
    37. case sort:
    38. SorContact(&con); //实现排序功能的函数
    39. break ;
    40. case ext :
    41. printf("程序结束\n");
    42. break;
    43. default :
    44. printf("输入有误请重新输入\n");
    45. }
    46. }while(choice);
    47. return 0;
    48. }

    二、功能的实现:

    2.1、增加人数:

    1. void AddContact(Contact* con)
    2. {
    3. assert(con); //指针不能为空
    4. if(con->count==Max)
    5. {
    6. printf("存放已满\n");
    7. return ;
    8. }
    9. printf("请输入名字");
    10. scanf("%s",con->data[con->count].name);
    11. printf("请输入年龄");
    12. scanf("%d",&(con->data[con->count].age));
    13. printf("请输入性别");
    14. scanf("%s",con->data[con->count].sex);
    15. printf("请输入电话");
    16. scanf("%s",con->data[con->count].tele);
    17. printf("请输入地址");
    18. scanf("%s",con->data[con->count].addr);
    19. con->count++;
    20. printf("增加成功\n");
    21. }

    2.2、删除人数:

    1. static int xiaomafind(Contact* con,char name[])
    2. {
    3. assert(con);
    4. int i=0;
    5. for(i=0;icount;i++)
    6. {
    7. if(0==strcmp(con->data[i].name,name)) //通过名字来查找看是否存在 ,如果有返回下标。
    8. {
    9. return i;
    10. }
    11. else
    12. {
    13. return -1;
    14. }
    15. }
    16. void DelContanct(Contact* con)
    17. {
    18. char name[20]={0};
    19. printf("请输入你想删除的名字\n");
    20. assert(con);
    21. int i;
    22. scanf("%s",name);
    23. int pos=xiaomafind(con,name);
    24. if(pos==-1)
    25. {
    26. printf("没有你要删除的人\n");
    27. }
    28. else
    29. {
    30. for(i=pos;icount;i++)
    31. {
    32. con->data[i]=con->data[i+1]; //通过数组后一个覆盖前一个来完成删除功能
    33. }
    34. con->count--; //删除完成后人员减一
    35. printf("删除成功\n");
    36. }
    37. }

    2.3、查找:

    1. static int xiaomafind(Contact* con,char name[])
    2. {
    3. assert(con);
    4. int i=0;
    5. for(i=0;icount;i++)
    6. {
    7. if(0==strcmp(con->data[i].name,name))
    8. {
    9. return i;
    10. }
    11. }
    12. return -1;
    13. }
    14. void SerContact(Contact *con)
    15. {
    16. char name[20]={0};
    17. printf("请输入你想查找的名字\n");
    18. assert(con);
    19. scanf("%s",name);
    20. int find =xiaomafind(con,name);
    21. if(find==-1)
    22. {
    23. printf("没有你要查找的用户\n");
    24. }
    25. else{
    26. printf("查找成功\n");
    27. }
    28. }

    2.4、展示:

    1. void ShowContact( const Contact* con)
    2. {
    3. assert(con);
    4. int i;
    5. for(i=0;icount;i++)
    6. {
    7. printf("%2s\t%3d\t%5s\t%12s\t%30s\n",con->data[i].name,con->data[i].age,con->data[i].sex,con->data[i].tele,con->data[i].addr); //这里是为了打印好看
    8. }
    9. }

    2.5、排序(这里我是通过名字):

    ^ _ ^用qsort 函数发现很很容的都能进行排序,我们只需要告诉他我们的排序方法是什么就可以很容易的进行一下排序

    1. int cmp_stu_by_name(const void*e1,const void*e2)
    2. {
    3. return strcmp(((peoInfo*)e1)->name,((peoInfo*)e2)->name);
    4. } //如不不理解的看我指针讲解中回调函数
    5. void SorContact(Contact* con)
    6. {
    7. assert(con);
    8. qsort(con->data, con->count, sizeof(peoInfo), cmp_stu_by_name); //这里运用了一个c语言的库函数qsort函数,在指针进阶中我也讲述了如何使用,如果不会的可以看一下指针的进阶。
    9. }

    三、通讯录进阶(设置动态存储):

    💓动态版本只需要改三个位置就行啦

    3.1通讯录从静态改为动态:

    1. 静态版本
    2. //typedef struct Contact
    3. //{
    4. // peoInfo data[100]; //人员名单 结构体数组
    5. // int count; //人员数量
    6. //}Contact;
    7. 动态版本
    8. typedef struct Contact
    9. {
    10. peoInfo *data; //人员名单 结构体指针
    11. int count; //人员数量
    12. int capicity; //通讯录容量
    13. }Contact;

    3.2通讯录的初始化:

    1. //静态版本
    2. //void InitContact(Contact* con)
    3. //{
    4. // assert(con);
    5. // con->count=0; //初始人数为0
    6. // memset(con->data,0,sizeof(con->data)); //这是一个c语言的库函数把数组里面的数全部初始化为0
    7. //}
    8. //动态版本
    9. void InitContact(Contact* con)
    10. {
    11. assert(con);
    12. con->count=0;
    13. con->data=(PeoInf0*)calloc(3,sizeof(peoInfo*));
    14. if(con->data=NULL)
    15. {
    16. perrror("con->data"); //如果空指针打印错误原因
    17. }
    18. con->capicity=3; //这里我们设置的初始容量为3
    19. }

    3.3通讯录的增加需要判断是否满了:

    1. void AddContact(Contact* con)
    2. {
    3. assert(con); //指针不能为空
    4. if(con->count==con->capicity)
    5. {
    6. (peoInfo*)ptr=(PeoInfo*)realloc(con->data,(con->capicity)*2*sizeof(peoInfo)). //增加人数前先进行判断是否满了,如果满了扩大2倍。
    7. if(ptr==NULL)
    8. {
    9. perror("peoInof");
    10. return ;
    11. }
    12. else{
    13. con->data=ptr;
    14. }
    15. }
    16. printf("请输入名字");
    17. scanf("%s",con->data[con->count].name);
    18. printf("请输入年龄");
    19. scanf("%d",&(con->data[con->count].age));
    20. printf("请输入性别");
    21. scanf("%s",con->data[con->count].sex);
    22. printf("请输入电话");
    23. scanf("%s",con->data[con->count].tele);
    24. printf("请输入地址");
    25. scanf("%s",con->data[con->count].addr);
    26. con->count++;
    27. printf("增加成功\n");
    28. }

    ^ - ^这样我们就实现通讯录的动态化,就不怕通讯录人数会存满了。🤓

    四、文件的形式存储通讯录:

          前面的通讯录在程序结束后发现人员没有办法进行一个保存,而通过文件的形式我们可以很好的做到这一点,在结束之前进行对文件的保存,在再一次执行程序是输入保存的文件,但是文件操作也有自己的不足之处,后面用数据库来实现更好,当前我们先用文件的形式来进行实现数据的保存,而实现这个只需要在结束时对文件保存,在初始化的时候时文件信息流入。

    4.1人员信息的保存:

    1. void SaveContact(Contact*con)
    2. {
    3. assert(con);
    4. //通过二进制写的形式把人员数据写入当中
    5. FILE* pfWrite=fopen("Contact.txt","wb");
    6. if(pfWrite==NULL)
    7. {
    8. perror("SaveContact");
    9. return ;
    10. }
    11. int i=0;
    12. for(i=0;icount;i++)
    13. {
    14. fwrite(con->data+i,sizeof(peoInfo),1,pfWrite); //一次写一个人员
    15. }
    16. fclose(pfWrite);
    17. pfWrite=NULL;
    18. }

    4.2人员信息的流入:

    1. void LoadContact(Contact* con)
    2. {
    3. assert(con);
    4. FILE*pfread =fopen("/Users/mamenghao/Desktop/Contact.txt","rb"); //二进制来读取文件信息
    5. if(pfread==NULL)
    6. {
    7. perror("LoadContact");
    8. return ;
    9. }
    10. peoInfo tmp={0}; //创建一个人员结构体变量先做临时的保存
    11. while( fread(&tmp,sizeof(peoInfo),1,pfread)==1)
    12. {
    13. if(con->count==con->capicity)
    14. {
    15. peoInfo* ptr=(peoInfo*)realloc(con->data,((con->capicity)*2*sizeof(peoInfo)));
    16. if(ptr==NULL)
    17. {
    18. perror("peoInof");
    19. return ;
    20. }
    21. else{
    22. con->data=ptr;
    23. }
    24. } //在增加前先判断是否容量已经满了
    25. con->data[con->count]=tmp;
    26. con->count++;
    27. }
    28. fclose(pfread);
    29. pfread=NULL;
    30. }

    总结:

           通讯录本身实现它并不是特别的难,运用的知识就是顺序表的增删插改功能,而进一步的优化只需要大家熟练掌握文件的操作和动态内存存储就能很好的实现啦!!

    最后小马码文不易,觉得有帮助的话请多多支持哈^ _ ^

  • 相关阅读:
    SQL查询优化---单表使用索引及常见索引失效优化
    Vue 父子组件传参、插槽
    二叉树 | 二叉搜索树 | 二叉树的有序性 | leecode刷题笔记
    外包干了3个多月,技术退步明显。。。。。
    职场PUA:为什么你就不能逼自己一把呢?
    十一月组队学习报名啦!
    聊聊绩效考核:松松网编辑跑路了吗?
    蓝桥杯备赛第四篇(高级数据结构)
    认识nginx
    MyBatisPlus 日志的两个坑:生产环境不打日志、多数据源日志配置等
  • 原文地址:https://blog.csdn.net/m0_63177573/article/details/127858400