C语言——通讯录

简介: C语言——通讯录

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

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

#define  _CRT_SECURE_NO_WARNINGS 1
#include "Contact.h"
int CheakCapacity(Contact* ps);
void LoadContact(Contact* ps)
{
  FILE* pf = fopen("Contact.dat", "rb");
  if (pf == NULL)
  {
    perror("LoadContact");
    return;
  }
  PeoInfo tmp = { 0 };
  while (fread(&tmp,sizeof(PeoInfo),1,pf))
  {
    CheakCapacity(ps);
    ps->date[ps->size] = tmp;
    ps->size++;
  }
  fclose(pf);
  pf = NULL;
}
void InitContact(Contact* ps)
{
  assert(ps);
  ps->date = NULL;
  ps->size = ps->capacity = 0;
  LoadContact(ps);
}
int CheakCapacity(Contact* ps)
{
  assert(ps);
  if (ps->capacity == ps->size)
  {
    int newcapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
    PeoInfo* tmp = (PeoInfo*)realloc(ps->date, newcapacity * sizeof(PeoInfo));
    if (tmp == NULL)
    {
      perror("CheakCapacity");
      return 0;
    }
    else
    {
      ps->date = tmp;
      ps->capacity = newcapacity;
      printf("增容成功\n");
      return 1;
    }
  }
  return 1;
}
void AddContact(Contact* ps)
{
  assert(ps);
  if (CheakCapacity(ps) == 0)
  {
    return;
  }
  printf("请输入增加的姓名:>");
  scanf("%s", ps->date[ps->size].name);
  printf("请输入增加的年龄:>");
  scanf("%d", &ps->date[ps->size].age);
  printf("请输入增加的性别:>");
  scanf("%s", ps->date[ps->size].sex);
  printf("请输入增加的电话:>");
  scanf("%s", ps->date[ps->size].tele);
  printf("请输入增加的地址:>");
  scanf("%s", ps->date[ps->size].addr);
  ps->size++;
  printf("增加成功\n");
}
void ShowContact(Contact* ps)
{
  assert(ps);
  printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
  for (int i =0; i<ps->size; i++)
  {
    printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
      ps->date[i].name,
      ps->date[i].age,
      ps->date[i].sex,
      ps->date[i].tele,
      ps->date[i].addr
      );
  }
}
int FindByName(const Contact* ps,char name[])
{
  int i = 0;
  for (i =0; i<ps->size; i++)
  {
    if (strcmp(ps->date[i].name,name)==0)
    {
      return i;
    }
  }
  return -1;
}
void DeleteContact(Contact* ps)
{
  assert(ps);
  if (ps->size == 0)
  {
    printf("通讯录为空,无法删除\n");
    return;
  }
  char name[20] = { 0 };
  printf("请输入你要删除的姓名:>");
  scanf("%s", name);
  int ret = FindByName(ps, name);
  if (ret == -1)
  {
    printf("要删除的人不存在\n");
    return;
  }
  for (int i =ret; i<ps->size; i++)
  {
    ps->date[i] = ps->date[i + 1];
  }
  ps->size--;
  printf("删除成功\n");
}
void SearchContact(Contact* ps)
{
  assert(ps);
  char name[20] = { 0 };
  printf("请输入你要删除的姓名:>");
  scanf("%s", name);
  int pos = FindByName(ps, name);
  if (pos == -1)
  {
    printf("要查找的人不存在\n");
    return;
  }
  printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
    ps->date[pos].name,
    ps->date[pos].age,
    ps->date[pos].sex,
    ps->date[pos].tele,
    ps->date[pos].addr
  );
}
void ModifyContact(Contact* ps)
{
  assert(ps);
  char name[20] = { 0 };
  printf("请输入你要修改的姓名:>");
  scanf("%s", name);
  int pos = FindByName(ps, name);
  if (pos == -1)
  {
    printf("要修改的人不存在\n");
    return;
  }
  printf("请输入修改的姓名:>");
  scanf("%s", ps->date[pos].name);
  printf("请输入修改的年龄:>");
  scanf("%d", &ps->date[pos].age);
  printf("请输入修改的性别:>");
  scanf("%s", ps->date[pos].sex);
  printf("请输入修改的电话:>");
  scanf("%s", ps->date[pos].tele);
  printf("请输入修改的地址:>");
  scanf("%s", ps->date[pos].addr);
}
void SortContact(Contact* ps)
{
  int i = 0;
  int j = 0;
  for (i=0; i<ps->size-1; i++)
  {
    for (j =0; j<ps->size-1-i; j++)
    {
      if (strcmp(ps->date[j].name, ps->date[j+1].name) > 0)
      {
        PeoInfo tmp = ps->date[j];
        ps->date[j] = ps->date[j + 1];
        ps->date[j + 1] = tmp;
      }
    }
  }
  printf("排序成功\n");
}
void DestoryContact(Contact* ps)
{
  free(ps->date);
  ps->date = NULL;
  ps->capacity = ps->size = 0;
}
void SaveContact(Contact* ps)
{
  FILE* pf = fopen("Contact.dat", "wb");
  if (pf == NULL)
  {
    perror("SaveContact");
    return;
  }
  for (int i =0; i<ps->size; i++)
  {
    fwrite(ps->date+ i, sizeof(PeoInfo), 1, pf);
  }
  fclose(pf);
  pf = NULL;
}
#define  _CRT_SECURE_NO_WARNINGS 1
#include "Contact.h"
void menu()
{
  printf("**************************** *************\n");
  printf("*********** 1.add     2.del **************\n");
  printf("*********** 3.search  4.modify ***********\n");
  printf("*********** 5.show    6.sort *************\n");
  printf("*********** 0.exit  ********* ************\n");
}
void Text()
{
  int input = 0;
  Contact con;
  InitContact(&con);
  do 
  {
    menu();
    printf("请输入你的选择:>");
    scanf("%d", &input);
    switch (input)
    {
    case ADD:
      AddContact(&con);
      break;
    case DEL:
      DeleteContact(&con);
      break;
    case SEARCH:
      SearchContact(&con);
      break;
    case MODIFY:
      ModifyContact(&con);
      break;
    case SHOW:
      ShowContact(&con);
      break;
    case SORT:
      SortContact(&con);
      break;
    case EXIT:
      SaveContact(&con);
      DestoryContact(&con);
      printf("退出通讯录成功\n");
      break;
    default:
      printf("你选择的有误,请重新输入\n");
      break;
    }
  } while (input);
}
int main()
{
  Text();
  return 0;
}

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


    • 我们要实现这个功能,首先我们来看下面的代码,首先我们应该先选择,这里我选择了do while的语句,要实现这个功能,我们就用到了menu这个函数来打印菜单,其实很简单,我就用了printf函数来实现。
    void Text()
    {
      int input = 0;
      Contact con;
      InitContact(&con);
      do 
      {
        menu();
        printf("请输入你的选择:>");
        scanf("%d", &input);
        switch (input)
        {
        case ADD:
          AddContact(&con);
          break;
        case DEL:
          DeleteContact(&con);
          break;
        case SEARCH:
          SearchContact(&con);
          break;
        case MODIFY:
          ModifyContact(&con);
          break;
        case SHOW:
          ShowContact(&con);
          break;
        case SORT:
          SortContact(&con);
          break;
        case EXIT:
          SaveContact(&con);
          DestoryContact(&con);
          printf("退出通讯录成功\n");
          break;
        default:
          printf("你选择的有误,请重新输入\n");
          break;
        }
      } while (input);
    }
    void menu()
    {
      printf("**************************** *************\n");
      printf("*********** 1.add     2.del **************\n");
      printf("*********** 3.search  4.modify ***********\n");
      printf("*********** 5.show    6.sort *************\n");
      printf("*********** 0.exit  ********* ************\n");
    }

    我们直接来实现通讯录的基本功能,我先用struct来定义了一个人的基本信息,然后再用struct来包含人的基本信息,但是我还加上了size和capcacity,size是用来记录存储了多少个人,而capacity是用来说明有多少个空间

    typedef struct PeoInfo
    {
      char name[MAX_NAME];
      int age;
      char sex[MAX_SEX];
      char tele[MAX_TELE];
      char addr[MAX_ADDR];
    }PeoInfo;
    typedef struct Contact
    {
      PeoInfo *date;
      int size;
      int capacity;
    }Contact;

    我们一开始用通讯录不要忘了要初始化,我在这里是把ps->date指向的空间置位了NULL,size和capacity初始化为0,因为我们是要弄一个动态的通讯录,所以我们特意用结构体的指针date来设计。

    我们实现一个通讯录,我们先设想一个场景,如果你的手机关机了,重启后是不是通讯录里面的信息还是存在的,所以我们也要实现这样的功能。

    • 我在这里是用到了文件操作,我先创建了结构体的tmp临时变量,我用fread来操作,如果freaed的返回值不是0,我们就将数据拷贝到tmp中。
    while (fread(&tmp,sizeof(PeoInfo),1,pf))
      {
        CheakCapacity(ps);
        ps->date[ps->size] = tmp;
        ps->size++;
      }
    • 下面是这个功能的全部代码
    void LoadContact(Contact* ps)
    {
      FILE* pf = fopen("Contact.dat", "rb");
      if (pf == NULL)
      {
        perror("LoadContact");
        return;
      }
      PeoInfo tmp = { 0 };
      while (fread(&tmp,sizeof(PeoInfo),1,pf))
      {
        CheakCapacity(ps);
        ps->date[ps->size] = tmp;
        ps->size++;
      }
      fclose(pf);
      pf = NULL;
    }
    void InitContact(Contact* ps)
    {
      assert(ps);
      ps->date = NULL;
      ps->size = ps->capacity = 0;
      LoadContact(ps);
    }

    接着我们来看第二个功能,增加人的信息,我们在一开始增加信息的时候,要想到如果空间满了的话,就要考虑扩容。所以,我一开始就判断是否要扩容,因为一开始size和capacity都是0,所以一开始就要扩容,我是malloc了一个空间,如果满了的话,我就扩二倍。最后通过返回值来判断是否扩容成功了。

    int CheakCapacity(Contact* ps)
    {
      assert(ps);
      if (ps->capacity == ps->size)
      {
        int newcapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
        PeoInfo* tmp = (PeoInfo*)realloc(ps->date, newcapacity * sizeof(PeoInfo));
        if (tmp == NULL)
        {
          perror("CheakCapacity");
          return 0;
        }
        else
        {
          ps->date = tmp;
          ps->capacity = newcapacity;
          printf("增容成功\n");
          return 1;
        }
      }
      return 1;
    }
    • 然后,接着实现add函数,其实很简单,我们一开始的size是0,所以每当我们增加一个信息,ps->size就要++,而ps->date指向的就是人信息的那片空间,ps->date【ps->size】后面再加上我们要增加的信息,就完成了我们add函数的功能。
    void AddContact(Contact* ps)
    {
      assert(ps);
      if (CheakCapacity(ps) == 0)
      {
        return;
      }
      printf("请输入增加的姓名:>");
      scanf("%s", ps->date[ps->size].name);
      printf("请输入增加的年龄:>");
      scanf("%d", &ps->date[ps->size].age);
      printf("请输入增加的性别:>");
      scanf("%s", ps->date[ps->size].sex);
      printf("请输入增加的电话:>");
      scanf("%s", ps->date[ps->size].tele);
      printf("请输入增加的地址:>");
      scanf("%s", ps->date[ps->size].addr);
      ps->size++;
      printf("增加成功\n");
    }
    • 第二个删除的功能,我的思想就是先创建一个数组,然后用数组和通讯录中名字相比较看是否相等。然后返回要删除的下标
    int FindByName(const Contact* ps,char name[])
    {
      int i = 0;
      for (i =0; i<ps->size; i++)
      {
        if (strcmp(ps->date[i].name,name)==0)
        {
          return i;
        }
      }
      return -1;
    }
    • 最后到删除的操作就是后面往前面移动,然后ps->size--就可以了。
    void DeleteContact(Contact* ps)
    {
      assert(ps);
      if (ps->size == 0)
      {
        printf("通讯录为空,无法删除\n");
        return;
      }
      char name[20] = { 0 };
      printf("请输入你要删除的姓名:>");
      scanf("%s", name);
      int ret = FindByName(ps, name);
      if (ret == -1)
      {
        printf("要删除的人不存在\n");
        return;
      }
      for (int i =ret; i<ps->size; i++)
      {
        ps->date[i] = ps->date[i + 1];
      }
      ps->size--;
      printf("删除成功\n");
    }
    • 第三个search功能的实现,也是和删除差不多的操作,不过我们是查找功能,所以我们最后是直接printf来打印出我们要查找的信息,这里我也用到了FindByName函数的复用。
    void SearchContact(Contact* ps)
    {
      assert(ps);
      char name[20] = { 0 };
      printf("请输入你要删除的姓名:>");
      scanf("%s", name);
      int pos = FindByName(ps, name);
      if (pos == -1)
      {
        printf("要查找的人不存在\n");
        return;
      }
      printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
        ps->date[pos].name,
        ps->date[pos].age,
        ps->date[pos].sex,
        ps->date[pos].tele,
        ps->date[pos].addr
      );
    }
    • 第四个修改的功能,因为我们也要找到要修改的下标,所以我用到了函数的复用,我们只要在要修改的下标中重新输入自己想修改的值就可以了。
    void ModifyContact(Contact* ps)
    {
      assert(ps);
      char name[20] = { 0 };
      printf("请输入你要修改的姓名:>");
      scanf("%s", name);
      int pos = FindByName(ps, name);
      if (pos == -1)
      {
        printf("要修改的人不存在\n");
        return;
      }
      printf("请输入修改的姓名:>");
      scanf("%s", ps->date[pos].name);
      printf("请输入修改的年龄:>");
      scanf("%d", &ps->date[pos].age);
      printf("请输入修改的性别:>");
      scanf("%s", ps->date[pos].sex);
      printf("请输入修改的电话:>");
      scanf("%s", ps->date[pos].tele);
      printf("请输入修改的地址:>");
      scanf("%s", ps->date[pos].addr);
    }

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

    • 第五个函数:就是我们要展示信息,我们直接用for循环遍历一遍就行了,只不过我们为了好看一点,用到了左对齐,至于长度可以根据你来实现。最后的效果就是下面这样。
    void ShowContact(Contact* ps)
    {
      assert(ps);
      printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
      for (int i =0; i<ps->size; i++)
      {
        printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
          ps->date[i].name,
          ps->date[i].age,
          ps->date[i].sex,
          ps->date[i].tele,
          ps->date[i].addr
          );
      }
    }
    • 第六个函数:我们要排序名字的大小,这里我直接用了冒泡排序,可以直接把名字的大小排出来。只不过这里的时间复杂度是O(N^2),效率是很低的,也可以用快排来实现,效率可以更高一点
    void SortContact(Contact* ps)
    {
      int i = 0;
      int j = 0;
      for (i=0; i<ps->size-1; i++)
      {
        for (j =0; j<ps->size-1-i; j++)
        {
          if (strcmp(ps->date[j].name, ps->date[j+1].name) > 0)
          {
            PeoInfo tmp = ps->date[j];
            ps->date[j] = ps->date[j + 1];
            ps->date[j + 1] = tmp;
          }
        }
      }
      printf("排序成功\n");
    }
    • 因为我们是动态来实现通讯录的,所以我用到了malloc,在最后退出的时候,也要free掉开辟的空间,我用了一个DestoryContact函数来实现,。
    void DestoryContact(Contact* ps)
    {
      free(ps->date);
      ps->date = NULL;
      ps->capacity = ps->size = 0;
    }

    最后一个函数:是否想过这样的一个问题,在我们输入信息的时候,如果我们退出,信息在下一次打开时还保存着呢???经过学习,我发现文件操作就可以来实现它 。

    我是用了fopen来打开一个二进制的文件Contact.dat,用到了for循环fwrite来把已经存在的信息保存在文件流中(也就是这个文件中),又因为一开始初始化的时候,要把文件的信息录进去,这样,我们就实现了信息的保存。

    void SaveContact(Contact* ps)
    {
      FILE* pf = fopen("Contact.dat", "wb");
      if (pf == NULL)
      {
        perror("SaveContact");
        return;
      }
      for (int i =0; i<ps->size; i++)
      {
        fwrite(ps->date+ i, sizeof(PeoInfo), 1, pf);
      }
      fclose(pf);
      pf = NULL;
    }


    相关文章
    |
    1月前
    |
    存储 C语言
    探索C语言数据结构:利用顺序表完成通讯录的实现
    本文介绍了如何使用C语言中的顺序表数据结构实现一个简单的通讯录,包括初始化、添加、删除、查找和保存联系人信息的操作,以及自定义结构体用于存储联系人详细信息。
    19 2
    |
    1月前
    |
    存储 C语言
    手把手教你用C语言实现通讯录管理系统
    手把手教你用C语言实现通讯录管理系统
    |
    6月前
    |
    C语言
    C语言——通讯录系统—基于 VS2022
    C语言——通讯录系统—基于 VS2022
    |
    3月前
    |
    存储 搜索推荐 算法
    【C语言】C语言—通讯录管理系统(源码)【独一无二】
    【C语言】C语言—通讯录管理系统(源码)【独一无二】
    |
    3月前
    |
    存储 数据可视化 C语言
    【C语言】C语言 手机通讯录系统的设计 (源码+数据+论文)【独一无二】
    【C语言】C语言 手机通讯录系统的设计 (源码+数据+论文)【独一无二】
    |
    5月前
    |
    机器学习/深度学习 搜索推荐 程序员
    C语言实现个人通讯录(功能优化)-2
    C语言实现个人通讯录(功能优化)
    C语言实现个人通讯录(功能优化)-2
    |
    5月前
    |
    存储 C语言 索引
    C语言实现个人通讯录(功能优化)-1
    C语言实现个人通讯录(功能优化)
    C语言实现个人通讯录(功能优化)-1
    |
    5月前
    |
    C语言
    C语言学习记录——通讯录(静态内存)
    C语言学习记录——通讯录(静态内存)
    32 2
    |
    6月前
    |
    存储 C语言
    C语言实现通讯录
    C语言实现通讯录
    40 2
    |
    6月前
    |
    存储 C语言
    C语言实验-动态顺序表实现简易通讯录(二)
    在这个C语言实验中,你将实现一个简单的通讯录,它使用动态顺序表来存储联系人信息。
    51 2