• 通讯录(纯C语言实现)


    相信大家都有过通讯录,今天我来带大家实现以下最简单的通讯录,通过本篇文章,相信可以让大家对C语言有进一步的认识。

    话不多说,我们先放函数的实现

    1. #define _CRT_SECURE_NO_WARNINGS 1
    2. #include "Contact.h"
    3. int CheakCapacity(Contact* ps);
    4. void LoadContact(Contact* ps)
    5. {
    6. FILE* pf = fopen("Contact.dat", "rb");
    7. if (pf == NULL)
    8. {
    9. perror("LoadContact");
    10. return;
    11. }
    12. PeoInfo tmp = { 0 };
    13. while (fread(&tmp,sizeof(PeoInfo),1,pf))
    14. {
    15. CheakCapacity(ps);
    16. ps->date[ps->size] = tmp;
    17. ps->size++;
    18. }
    19. fclose(pf);
    20. pf = NULL;
    21. }
    22. void InitContact(Contact* ps)
    23. {
    24. assert(ps);
    25. ps->date = NULL;
    26. ps->size = ps->capacity = 0;
    27. LoadContact(ps);
    28. }
    29. int CheakCapacity(Contact* ps)
    30. {
    31. assert(ps);
    32. if (ps->capacity == ps->size)
    33. {
    34. int newcapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
    35. PeoInfo* tmp = (PeoInfo*)realloc(ps->date, newcapacity * sizeof(PeoInfo));
    36. if (tmp == NULL)
    37. {
    38. perror("CheakCapacity");
    39. return 0;
    40. }
    41. else
    42. {
    43. ps->date = tmp;
    44. ps->capacity = newcapacity;
    45. printf("增容成功\n");
    46. return 1;
    47. }
    48. }
    49. return 1;
    50. }
    51. void AddContact(Contact* ps)
    52. {
    53. assert(ps);
    54. if (CheakCapacity(ps) == 0)
    55. {
    56. return;
    57. }
    58. printf("请输入增加的姓名:>");
    59. scanf("%s", ps->date[ps->size].name);
    60. printf("请输入增加的年龄:>");
    61. scanf("%d", &ps->date[ps->size].age);
    62. printf("请输入增加的性别:>");
    63. scanf("%s", ps->date[ps->size].sex);
    64. printf("请输入增加的电话:>");
    65. scanf("%s", ps->date[ps->size].tele);
    66. printf("请输入增加的地址:>");
    67. scanf("%s", ps->date[ps->size].addr);
    68. ps->size++;
    69. printf("增加成功\n");
    70. }
    71. void ShowContact(Contact* ps)
    72. {
    73. assert(ps);
    74. printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
    75. for (int i =0; i<ps->size; i++)
    76. {
    77. printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
    78. ps->date[i].name,
    79. ps->date[i].age,
    80. ps->date[i].sex,
    81. ps->date[i].tele,
    82. ps->date[i].addr
    83. );
    84. }
    85. }
    86. int FindByName(const Contact* ps,char name[])
    87. {
    88. int i = 0;
    89. for (i =0; i<ps->size; i++)
    90. {
    91. if (strcmp(ps->date[i].name,name)==0)
    92. {
    93. return i;
    94. }
    95. }
    96. return -1;
    97. }
    98. void DeleteContact(Contact* ps)
    99. {
    100. assert(ps);
    101. if (ps->size == 0)
    102. {
    103. printf("通讯录为空,无法删除\n");
    104. return;
    105. }
    106. char name[20] = { 0 };
    107. printf("请输入你要删除的姓名:>");
    108. scanf("%s", name);
    109. int ret = FindByName(ps, name);
    110. if (ret == -1)
    111. {
    112. printf("要删除的人不存在\n");
    113. return;
    114. }
    115. for (int i =ret; i<ps->size; i++)
    116. {
    117. ps->date[i] = ps->date[i + 1];
    118. }
    119. ps->size--;
    120. printf("删除成功\n");
    121. }
    122. void SearchContact(Contact* ps)
    123. {
    124. assert(ps);
    125. char name[20] = { 0 };
    126. printf("请输入你要删除的姓名:>");
    127. scanf("%s", name);
    128. int pos = FindByName(ps, name);
    129. if (pos == -1)
    130. {
    131. printf("要查找的人不存在\n");
    132. return;
    133. }
    134. printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
    135. ps->date[pos].name,
    136. ps->date[pos].age,
    137. ps->date[pos].sex,
    138. ps->date[pos].tele,
    139. ps->date[pos].addr
    140. );
    141. }
    142. void ModifyContact(Contact* ps)
    143. {
    144. assert(ps);
    145. char name[20] = { 0 };
    146. printf("请输入你要修改的姓名:>");
    147. scanf("%s", name);
    148. int pos = FindByName(ps, name);
    149. if (pos == -1)
    150. {
    151. printf("要修改的人不存在\n");
    152. return;
    153. }
    154. printf("请输入修改的姓名:>");
    155. scanf("%s", ps->date[pos].name);
    156. printf("请输入修改的年龄:>");
    157. scanf("%d", &ps->date[pos].age);
    158. printf("请输入修改的性别:>");
    159. scanf("%s", ps->date[pos].sex);
    160. printf("请输入修改的电话:>");
    161. scanf("%s", ps->date[pos].tele);
    162. printf("请输入修改的地址:>");
    163. scanf("%s", ps->date[pos].addr);
    164. }
    165. void SortContact(Contact* ps)
    166. {
    167. int i = 0;
    168. int j = 0;
    169. for (i=0; i<ps->size-1; i++)
    170. {
    171. for (j =0; j<ps->size-1-i; j++)
    172. {
    173. if (strcmp(ps->date[j].name, ps->date[j+1].name) > 0)
    174. {
    175. PeoInfo tmp = ps->date[j];
    176. ps->date[j] = ps->date[j + 1];
    177. ps->date[j + 1] = tmp;
    178. }
    179. }
    180. }
    181. printf("排序成功\n");
    182. }
    183. void DestoryContact(Contact* ps)
    184. {
    185. free(ps->date);
    186. ps->date = NULL;
    187. ps->capacity = ps->size = 0;
    188. }
    189. void SaveContact(Contact* ps)
    190. {
    191. FILE* pf = fopen("Contact.dat", "wb");
    192. if (pf == NULL)
    193. {
    194. perror("SaveContact");
    195. return;
    196. }
    197. for (int i =0; i<ps->size; i++)
    198. {
    199. fwrite(ps->date+ i, sizeof(PeoInfo), 1, pf);
    200. }
    201. fclose(pf);
    202. pf = NULL;
    203. }

    1. #define _CRT_SECURE_NO_WARNINGS 1
    2. #include "Contact.h"
    3. void menu()
    4. {
    5. printf("**************************** *************\n");
    6. printf("*********** 1.add 2.del **************\n");
    7. printf("*********** 3.search 4.modify ***********\n");
    8. printf("*********** 5.show 6.sort *************\n");
    9. printf("*********** 0.exit ********* ************\n");
    10. }
    11. void Text()
    12. {
    13. int input = 0;
    14. Contact con;
    15. InitContact(&con);
    16. do
    17. {
    18. menu();
    19. printf("请输入你的选择:>");
    20. scanf("%d", &input);
    21. switch (input)
    22. {
    23. case ADD:
    24. AddContact(&con);
    25. break;
    26. case DEL:
    27. DeleteContact(&con);
    28. break;
    29. case SEARCH:
    30. SearchContact(&con);
    31. break;
    32. case MODIFY:
    33. ModifyContact(&con);
    34. break;
    35. case SHOW:
    36. ShowContact(&con);
    37. break;
    38. case SORT:
    39. SortContact(&con);
    40. break;
    41. case EXIT:
    42. SaveContact(&con);
    43. DestoryContact(&con);
    44. printf("退出通讯录成功\n");
    45. break;
    46. default:
    47. printf("你选择的有误,请重新输入\n");
    48. break;
    49. }
    50. } while (input);
    51. }
    52. int main()
    53. {
    54. Text();
    55. return 0;
    56. }

     是不是看到这里会感到很害怕??不用怕,跟着我的思路,你也可以实现它,我带着你一步一步实现每一个功能

    •  我们要实现这个功能,首先我们来看下面的代码,首先我们应该先选择,这里我选择了do while的语句,要实现这个功能,我们就用到了menu这个函数来打印菜单,其实很简单,我就用了printf函数来实现。
    1. void Text()
    2. {
    3. int input = 0;
    4. Contact con;
    5. InitContact(&con);
    6. do
    7. {
    8. menu();
    9. printf("请输入你的选择:>");
    10. scanf("%d", &input);
    11. switch (input)
    12. {
    13. case ADD:
    14. AddContact(&con);
    15. break;
    16. case DEL:
    17. DeleteContact(&con);
    18. break;
    19. case SEARCH:
    20. SearchContact(&con);
    21. break;
    22. case MODIFY:
    23. ModifyContact(&con);
    24. break;
    25. case SHOW:
    26. ShowContact(&con);
    27. break;
    28. case SORT:
    29. SortContact(&con);
    30. break;
    31. case EXIT:
    32. SaveContact(&con);
    33. DestoryContact(&con);
    34. printf("退出通讯录成功\n");
    35. break;
    36. default:
    37. printf("你选择的有误,请重新输入\n");
    38. break;
    39. }
    40. } while (input);
    41. }
    1. void menu()
    2. {
    3. printf("**************************** *************\n");
    4. printf("*********** 1.add 2.del **************\n");
    5. printf("*********** 3.search 4.modify ***********\n");
    6. printf("*********** 5.show 6.sort *************\n");
    7. printf("*********** 0.exit ********* ************\n");
    8. }

    • 我们直接来实现通讯录的基本功能,我先用struct来定义了一个人的基本信息,然后再用struct来包含人的基本信息,但是我还加上了size和capcacity,size是用来记录存储了多少个人,而capacity是用来说明有多少个空间
      1. typedef struct PeoInfo
      2. {
      3. char name[MAX_NAME];
      4. int age;
      5. char sex[MAX_SEX];
      6. char tele[MAX_TELE];
      7. char addr[MAX_ADDR];
      8. }PeoInfo;
      9. typedef struct Contact
      10. {
      11. PeoInfo *date;
      12. int size;
      13. int capacity;
      14. }Contact;

    •  我们一开始用通讯录不要忘了要初始化,我在这里是把ps->date指向的空间置位了NULL,size和capacity初始化为0,因为我们是要弄一个动态的通讯录,所以我们特意用结构体的指针date来设计。
    • 我们实现一个通讯录,我们先设想一个场景,如果你的手机关机了,重启后是不是通讯录里面的信息还是存在的,所以我们也要实现这样的功能。
    • 我在这里是用到了文件操作,我先创建了结构体的tmp临时变量,我用fread来操作,如果freaed的返回值不是0,我们就将数据拷贝到tmp中。
      1. while (fread(&tmp,sizeof(PeoInfo),1,pf))
      2. {
      3. CheakCapacity(ps);
      4. ps->date[ps->size] = tmp;
      5. ps->size++;
      6. }

      下面是这个功能的全部代码

      1. void LoadContact(Contact* ps)
      2. {
      3. FILE* pf = fopen("Contact.dat", "rb");
      4. if (pf == NULL)
      5. {
      6. perror("LoadContact");
      7. return;
      8. }
      9. PeoInfo tmp = { 0 };
      10. while (fread(&tmp,sizeof(PeoInfo),1,pf))
      11. {
      12. CheakCapacity(ps);
      13. ps->date[ps->size] = tmp;
      14. ps->size++;
      15. }
      16. fclose(pf);
      17. pf = NULL;
      18. }
      19. void InitContact(Contact* ps)
      20. {
      21. assert(ps);
      22. ps->date = NULL;
      23. ps->size = ps->capacity = 0;
      24. LoadContact(ps);
      25. }

    •  接着我们来看第二个功能,增加人的信息,我们在一开始增加信息的时候,要想到如果空间满了的话,就要考虑扩容。所以,我一开始就判断是否要扩容,因为一开始size和capacity都是0,所以一开始就要扩容,我是malloc了一个空间,如果满了的话,我就扩二倍。最后通过返回值来判断是否扩容成功了。
    1. int CheakCapacity(Contact* ps)
    2. {
    3. assert(ps);
    4. if (ps->capacity == ps->size)
    5. {
    6. int newcapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
    7. PeoInfo* tmp = (PeoInfo*)realloc(ps->date, newcapacity * sizeof(PeoInfo));
    8. if (tmp == NULL)
    9. {
    10. perror("CheakCapacity");
    11. return 0;
    12. }
    13. else
    14. {
    15. ps->date = tmp;
    16. ps->capacity = newcapacity;
    17. printf("增容成功\n");
    18. return 1;
    19. }
    20. }
    21. return 1;
    22. }
    • 然后,接着实现add函数,其实很简单,我们一开始的size是0,所以每当我们增加一个信息,ps->size就要++,而ps->date指向的就是人信息的那片空间,ps->date【ps->size】后面再加上我们要增加的信息,就完成了我们add函数的功能。
    1. void AddContact(Contact* ps)
    2. {
    3. assert(ps);
    4. if (CheakCapacity(ps) == 0)
    5. {
    6. return;
    7. }
    8. printf("请输入增加的姓名:>");
    9. scanf("%s", ps->date[ps->size].name);
    10. printf("请输入增加的年龄:>");
    11. scanf("%d", &ps->date[ps->size].age);
    12. printf("请输入增加的性别:>");
    13. scanf("%s", ps->date[ps->size].sex);
    14. printf("请输入增加的电话:>");
    15. scanf("%s", ps->date[ps->size].tele);
    16. printf("请输入增加的地址:>");
    17. scanf("%s", ps->date[ps->size].addr);
    18. ps->size++;
    19. printf("增加成功\n");
    20. }


    •  第二个删除的功能,我的思想就是先创建一个数组,然后用数组和通讯录中名字相比较看是否相等。然后返回要删除的下标
    1. int FindByName(const Contact* ps,char name[])
    2. {
    3. int i = 0;
    4. for (i =0; i<ps->size; i++)
    5. {
    6. if (strcmp(ps->date[i].name,name)==0)
    7. {
    8. return i;
    9. }
    10. }
    11. return -1;
    12. }
    • 最后到删除的操作就是后面往前面移动,然后ps->size--就可以了。
    1. void DeleteContact(Contact* ps)
    2. {
    3. assert(ps);
    4. if (ps->size == 0)
    5. {
    6. printf("通讯录为空,无法删除\n");
    7. return;
    8. }
    9. char name[20] = { 0 };
    10. printf("请输入你要删除的姓名:>");
    11. scanf("%s", name);
    12. int ret = FindByName(ps, name);
    13. if (ret == -1)
    14. {
    15. printf("要删除的人不存在\n");
    16. return;
    17. }
    18. for (int i =ret; i<ps->size; i++)
    19. {
    20. ps->date[i] = ps->date[i + 1];
    21. }
    22. ps->size--;
    23. printf("删除成功\n");
    24. }


    •  第三个search功能的实现,也是和删除差不多的操作,不过我们是查找功能,所以我们最后是直接printf来打印出我们要查找的信息,这里我也用到了FindByName函数的复用。
    1. void SearchContact(Contact* ps)
    2. {
    3. assert(ps);
    4. char name[20] = { 0 };
    5. printf("请输入你要删除的姓名:>");
    6. scanf("%s", name);
    7. int pos = FindByName(ps, name);
    8. if (pos == -1)
    9. {
    10. printf("要查找的人不存在\n");
    11. return;
    12. }
    13. printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
    14. ps->date[pos].name,
    15. ps->date[pos].age,
    16. ps->date[pos].sex,
    17. ps->date[pos].tele,
    18. ps->date[pos].addr
    19. );
    20. }


    •  第四个修改的功能,因为我们也要找到要修改的下标,所以我用到了函数的复用,我们只要在要修改的下标中重新输入自己想修改的值就可以了。
    1. void ModifyContact(Contact* ps)
    2. {
    3. assert(ps);
    4. char name[20] = { 0 };
    5. printf("请输入你要修改的姓名:>");
    6. scanf("%s", name);
    7. int pos = FindByName(ps, name);
    8. if (pos == -1)
    9. {
    10. printf("要修改的人不存在\n");
    11. return;
    12. }
    13. printf("请输入修改的姓名:>");
    14. scanf("%s", ps->date[pos].name);
    15. printf("请输入修改的年龄:>");
    16. scanf("%d", &ps->date[pos].age);
    17. printf("请输入修改的性别:>");
    18. scanf("%s", ps->date[pos].sex);
    19. printf("请输入修改的电话:>");
    20. scanf("%s", ps->date[pos].tele);
    21. printf("请输入修改的地址:>");
    22. scanf("%s", ps->date[pos].addr);
    23. }


     是不是发现其实函数的实现很多都是相似的, 所以不用怕,接着往后看。

    •  第五个函数:就是我们要展示信息,我们直接用for循环遍历一遍就行了,只不过我们为了好看一点,用到了左对齐,至于长度可以根据你来实现。最后的效果就是下面这样。

    1. void ShowContact(Contact* ps)
    2. {
    3. assert(ps);
    4. printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
    5. for (int i =0; i<ps->size; i++)
    6. {
    7. printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
    8. ps->date[i].name,
    9. ps->date[i].age,
    10. ps->date[i].sex,
    11. ps->date[i].tele,
    12. ps->date[i].addr
    13. );
    14. }
    15. }


    •  第六个函数:我们要排序名字的大小,这里我直接用了冒泡排序,可以直接把名字的大小排出来。只不过这里的时间复杂度是O(N^2),效率是很低的,也可以用快排来实现,效率可以更高一点
    1. void SortContact(Contact* ps)
    2. {
    3. int i = 0;
    4. int j = 0;
    5. for (i=0; i<ps->size-1; i++)
    6. {
    7. for (j =0; j<ps->size-1-i; j++)
    8. {
    9. if (strcmp(ps->date[j].name, ps->date[j+1].name) > 0)
    10. {
    11. PeoInfo tmp = ps->date[j];
    12. ps->date[j] = ps->date[j + 1];
    13. ps->date[j + 1] = tmp;
    14. }
    15. }
    16. }
    17. printf("排序成功\n");
    18. }


    •  因为我们是动态来实现通讯录的,所以我用到了malloc,在最后退出的时候,也要free掉开辟的空间,我用了一个DestoryContact函数来实现,。
      1. void DestoryContact(Contact* ps)
      2. {
      3. free(ps->date);
      4. ps->date = NULL;
      5. ps->capacity = ps->size = 0;
      6. }


    • 最后一个函数:是否想过这样的一个问题,在我们输入信息的时候,如果我们退出,信息在下一次打开时还保存着呢???经过学习,我发现文件操作就可以来实现它 。
    • 我是用了fopen来打开一个二进制的文件Contact.dat,用到了for循环fwrite来把已经存在的信息保存在文件流中(也就是这个文件中),又因为一开始初始化的时候,要把文件的信息录进去,这样,我们就实现了信息的保存。
    1. void SaveContact(Contact* ps)
    2. {
    3. FILE* pf = fopen("Contact.dat", "wb");
    4. if (pf == NULL)
    5. {
    6. perror("SaveContact");
    7. return;
    8. }
    9. for (int i =0; i<ps->size; i++)
    10. {
    11. fwrite(ps->date+ i, sizeof(PeoInfo), 1, pf);
    12. }
    13. fclose(pf);
    14. pf = NULL;
    15. }

  • 相关阅读:
    Android Native Thread
    SpringCloud-3-基础工程构建
    大顶堆的实现(基于数组存储的完全二叉树)
    缓存淘汰算法-LRU
    27.集合框架-Map接口及其子类和Collections类(3)[20220728]
    电商转化率这么抽象,到底是个啥?
    Java 实现分布式定时任务
    ctf-pikachu-fileupload
    Java学数据结构(4)——PriorityQueue(优先队列)& 二叉堆(binary heap)
    最新Next 14快速上手基础部分
  • 原文地址:https://blog.csdn.net/m0_72165281/article/details/131745871