【c语言进阶】常见的静态通讯录 中

简介: 【c语言进阶】常见的静态通讯录 中

👻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("排序成功");
}
目录
相关文章
|
2天前
|
存储 C语言
C语言进阶---------作业复习
C语言进阶---------作业复习
|
2天前
|
存储 Linux C语言
C语言进阶第十一节 --------程序环境和预处理(包含宏的解释)-2
C语言进阶第十一节 --------程序环境和预处理(包含宏的解释)
|
2天前
|
自然语言处理 Linux 编译器
C语言进阶第十一节 --------程序环境和预处理(包含宏的解释)-1
C语言进阶第十一节 --------程序环境和预处理(包含宏的解释)
|
2天前
|
存储 编译器 C语言
C语言进阶第十课 --------文件的操作-1
C语言进阶第十课 --------文件的操作
|
2天前
|
存储 程序员 C语言
C语言进阶第九课 --------动态内存管理-2
C语言进阶第九课 --------动态内存管理
|
2天前
|
编译器 C语言
C语言进阶第九课 --------动态内存管理-1
C语言进阶第九课 --------动态内存管理
|
2天前
|
C语言
C语言进阶第八课 --------通讯录的实现
C语言进阶第八课 --------通讯录的实现
|
2天前
|
存储 算法 C语言
C语言进阶:顺序表(数据结构基础) (以通讯录项目为代码练习)
C语言进阶:顺序表(数据结构基础) (以通讯录项目为代码练习)
|
2天前
通讯录(C语言版)
通讯录(C语言版)
|
2天前
|
存储 人工智能 搜索推荐
【C语言进阶】 假期测评①
【C语言进阶】 假期测评①
38 2