C语言实现通讯录 (附完整代码) 2

简介: C语言实现通讯录 (附完整代码)

🌼ShowContact

//显示所有联系人的信息
void ShowContact(const Contact* pc)//仅仅是显示,而不修改 所以用const限制修饰
{
  int i;
  for (i = 0; i < pc->sz; i++)
  {
    printf("姓名:%-20s\t", pc->data[i].name);//根据联系人结构体信息
    printf("年龄:%-4d\t", pc->data[i].age);
    printf("性别:%-5s\t", pc->data[i].sex);
    printf("电话:%-12s\t", pc->data[i].tele);
    printf("住址:%-13s\t", pc->data[i].addr);//统一左对齐
    printf("\n");
  }
}

但是,当显示的数据变多的时候,尽管是左对齐,还是不够美观。

接下来,我们采用打印标题的形式,更好的打印出数据。

//显示所有联系人的信息
void ShowContact(const Contact* pc)//仅仅是显示,而不修改 所以用const限制修饰
{
  int i;
  //打印列标题
  printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
  //打印数据
  for (i = 0; i < pc->sz; i++)
  {
    printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\t\n",
      pc->data[i].name,
      pc->data[i].age,
      pc->data[i].sex,
      pc->data[i].tele,
      pc->data[i].addr);
  }
}

🌼DelContact

//删除指定联系人
void DelContact(Contact* pc)
{
  int i = 0;
  char name[20];
  int pos = 0;
  int flag = 0;//输入要删除的人删除
  printf("请输入要删除人的名字:>");
  scanf("%s", name);//将要删除的人的名字放入name中(name本就是一个地址值)
  //查找有没有这个人
  for (i = 0; i < pc->sz; i ++)
  {
    if (strcmp(name, pc->data[i].name) == 0) //字符串比较函数
    {
      pos = i;//找到了,记下位置
      flag = 1;
    } 
  }
  if (flag == 0)
  {
    printf("要删除的人不存在\n");
    return;
  }
  //进行删除
  for (i = pos; i <pc->sz-1; i++)
  {
    pc->data[i] = pc->data[i + 1];
  }
  pc->sz--;
  printf("成功删除联系人!");
}

🌼进一步改进

当我们写后面的函数时,我们发现后面的查找函数 还是 修改函数 都需要像 删除函数一样 先在通讯录类进行查找。

为了更高效完成整个工程,提高效率,我们可以来写一个通过名字进行查找的函数FindByName

当我们写 删除 查找 修改函数时,首先可以直接调用这个函数。

int FindByName(Contact* pc, char name[])
{
  int flag = 0;
  int i = 0;
  int pos;
  for (i = 0; i < pc->sz; i++)
  {
    if (strcmp(name, pc->data[i].name) == 0) //字符串比较函数
    {
      pos = i;//找到了,记下位置
      return pos;
    }
  }
  if (flag == 0)
  {
    printf("要删除的人不存在\n");
    return -1;
  }
}

DelContact函数改进为:

//删除指定联系人
void DelContact(Contact* pc)
{
  int i = 0;
  char name[20];
  int pos = 0;
  int flag = 0;//输入要删除的人删除
  printf("请输入要删除人的名字:>");
  scanf("%s", name);//将要删除的人的名字放入name中(name本就是一个地址值)
  //查找有没有这个人
  if ((FindByName(pc, name) == -1))
  {
    printf("找不到要删除的联系人");
  }
  else
    pos = FindByName(pc, name); //记下位置
  //进行删除
  for (i = pos; i <pc->sz-1; i++)
  {
    pc->data[i] = pc->data[i + 1];
  }
  pc->sz--;
  printf("成功删除联系人!");
}

🌼SearchContact

//查找指定联系人
void SearchContact(const Contact* pc)
{
  int pos = 0;
  char name[20] = { 0 };
  printf("请输入要查找人的名字:>");
  scanf("%s", name);
  //查找有没有指定联系人
  if (FindByName(pc, name) == -1)
    printf("没有找到要查找的联系人\n");
  else
  {
    pos = FindByName(pc, name);
    printf("找到了!\n");
    printf("该联系人的下标为%d", pos);
    //打印数据
    printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
    printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\t\n",
      pc->data[pos].name,
      pc->data[pos].age,
      pc->data[pos].sex,
      pc->data[pos].tele,
      pc->data[pos].addr);
  }
}

🌼ModifyContact

//修改指定联系人
void ModifyContact(Contact* pc)
{
  int pos = 0;
  char name[20] = { 0 };
  printf("请输入要查找人的名字:>");
  scanf("%s", name);
  //查找有没有要修改的联系人
  if (FindByName(pc, name) == -1)
    printf("没有找到要修改的联系人\n");
  else 
  {
    printf("找到了要修改的联系人!\n");
    pos = FindByName(pc, name);
    //修改 (修改可以理解为再次录入一遍信息)
    printf("请输入名字:>");
    scanf("%s", pc->data[pos].name);
    printf("请输入年龄:>");
    scanf("%d", &(pc->data[pos].age));
    printf("请输入性别:>");
    scanf("%s", pc->data[pos].sex);
    printf("请输入电话:>");
    scanf("%s", pc->data[pos].tele);
    printf("请输入住址:>");
    scanf("%s", pc->data[pos].addr);
    printf("成功修改联系人!");
  }
}

🌼Sort Contact

//按照名字进行排序
//排序
int cmp_name(const void* p1, const void* p2)
{
  return strcmp(((PeoInfo*)p1)->name, ((PeoInfo*)p2)->name);
}
void SortContact(Contact* pc)
{
  int i = 0;
  //利用qsort 函数
  qsort(pc->data, pc->sz, sizeof((pc->data)[0]), cmp_name);
  //打印列标题
  printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
  //打印数据
  for (i = 0; i < pc->sz; i++)
  {
    printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\t\n",
      pc->data[i].name,
      pc->data[i].age,
      pc->data[i].sex,
      pc->data[i].tele,
      pc->data[i].addr);
  }
}

🎆🎆🎆完整代码

contact.h

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#define MAX 100
//声明结构体--保存人的信息
typedef struct PeoInfo
{
  char name[20];
  int age;
  char sex[5];//一个汉字占两个字符
  char tele[12];
  char addr[30];
}PeoInfo;
//声明结构体通讯录
typedef struct Contact
{
  PeoInfo data[MAX];
  int sz;
}Contact;
int FindByName(Contact* pc, char name[]);
void AddContact(Contact* pc);
void ShowContact(const Contact* pc);
void DelContact(Contact* pc);
void SearchContact( Contact* pc);
void SearchContact(Contact* pc);
void SortContact(Contact* pc);
enum OPTION
{
  EXIT,//0
  ADD,//1
  DEL,//2
  SEARCH,//3
  MODIFY,//4
  SHOW,//5
  SORT//6
};
//函数的声明--初始化通讯录
void InitContact(Contact* pc);

contact.c

#include "contact.h"
void InitContact(Contact* pc)
{
  memset(pc->data, 0, sizeof(pc->data));
  pc->sz = 0;
}
//增加联系人的信息
void AddContact(Contact* pc)
{
  //判断数组是否可以增添?
  if (pc->sz == 100)
  {
    printf("通讯录已满,无法添加\n");
    return;
  }
  //通讯录没满
  else
  {
    printf("请输入名字:>");
    scanf("%s", pc->data[pc->sz].name);//pc->data 指向的是结构体数组 数组名本身就是地址,不用用取地址符号
    printf("请输入年龄:>");
    scanf("%d", &(pc->data[pc->sz].age));
    printf("请输入性别:>");
    scanf("%s", pc->data[pc->sz].sex);
    printf("请输入电话:>");
    scanf("%s", pc->data[pc->sz].tele);
    printf("请输入住址:>");
    scanf("%s", pc->data[pc->sz].addr);
    pc->sz++;
    printf("增加联系人成功!");
  }
}
//显示所有联系人的信息
void ShowContact(const Contact* pc)//仅仅是显示,而不修改 所以用const限制修饰
{
  int i;
  //打印列标题
  printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
  //打印数据
  for (i = 0; i < pc->sz; i++)
  {
    printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\t\n",
      pc->data[i].name,
      pc->data[i].age,
      pc->data[i].sex,
      pc->data[i].tele,
      pc->data[i].addr);
  }
}
int FindByName(Contact* pc, char name[])
{
  int flag = 0;
  int i = 0;
  int pos;
  for (i = 0; i < pc->sz; i++)
  {
    if (strcmp(name, pc->data[i].name) == 0) //字符串比较函数
    {
      pos = i;//找到了,记下位置
      return pos;
    }
  }
  if (flag == 0)
  {
    printf("要删除的人不存在\n");
    return -1;
  }
}
//删除指定联系人
void DelContact(Contact* pc)
{
  int i = 0;
  char name[20];
  int pos = 0;
  int flag = 0;//输入要删除的人删除
  printf("请输入要删除人的名字:>");
  scanf("%s", name);//将要删除的人的名字放入name中(name本就是一个地址值)
  //查找有没有这个人
  if ((FindByName(pc, name) == -1))
  {
    printf("找不到要删除的联系人");
  }
  else
    pos = FindByName(pc, name); //记下位置
  //进行删除
  for (i = pos; i <pc->sz-1; i++)
  {
    pc->data[i] = pc->data[i + 1];
  }
  pc->sz--;
  printf("成功删除联系人!");
}
//查找指定联系人
void SearchContact( Contact* pc)
{
  int pos = 0;
  char name[20] = { 0 };
  printf("请输入要查找人的名字:>");
  scanf("%s", name);
  //查找有没有指定联系人
  if (FindByName(pc, name) == -1)
    printf("没有找到要查找的联系人\n");
  else
  {
    pos = FindByName(pc, name);
    printf("找到了!\n");
    printf("该联系人的下标为%d", pos);
    //打印数据
    printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
    printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\t\n",
      pc->data[pos].name,
      pc->data[pos].age,
      pc->data[pos].sex,
      pc->data[pos].tele,
      pc->data[pos].addr);
  }
}
//修改指定联系人
void ModifyContact(Contact* pc)
{
  int pos = 0;
  char name[20] = { 0 };
  printf("请输入要查找人的名字:>");
  scanf("%s", name);
  //查找有没有要修改的联系人
  if (FindByName(pc, name) == -1)
    printf("没有找到要修改的联系人\n");
  else 
  {
    printf("找到了要修改的联系人!\n");
    pos = FindByName(pc, name);
    //修改 (修改可以理解为再次录入一遍信息)
    printf("请输入名字:>");
    scanf("%s", pc->data[pos].name);
    printf("请输入年龄:>");
    scanf("%d", &(pc->data[pos].age));
    printf("请输入性别:>");
    scanf("%s", pc->data[pos].sex);
    printf("请输入电话:>");
    scanf("%s", pc->data[pos].tele);
    printf("请输入住址:>");
    scanf("%s", pc->data[pos].addr);
    printf("成功修改联系人!");
  }
}
//按照名字进行排序
//排序
int cmp_name(const void* p1, const void* p2)
{
  return strcmp(((PeoInfo*)p1)->name, ((PeoInfo*)p2)->name);
}
void SortContact(Contact* pc)
{
  int i = 0;
  //利用qsort 函数
  qsort(pc->data, pc->sz, sizeof((pc->data)[0]), cmp_name);
  //打印列标题
  printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
  //打印数据
  for (i = 0; i < pc->sz; i++)
  {
    printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\t\n",
      pc->data[i].name,
      pc->data[i].age,
      pc->data[i].sex,
      pc->data[i].tele,
      pc->data[i].addr);
  }
}

test.c

#include"contact.h"
void menu()
{
  printf("\n");
  printf("******************\n");
  printf("******请选择******\n");
  printf("******1.增添******\n");
  printf("******2.删除******\n");
  printf("******3.查找******\n");
  printf("******4.修改******\n");
  printf("******5.显示******\n");
  printf("******6.排序******\n");
  printf("******0.退出******\n");
}
void test()
{
  Contact con;  //定义结构体变量-通讯录
  InitContact(&con);//对通讯录进行初始化
  int input = 0;
  do
  {
    menu();
    scanf("%d", &input);
    switch (input)
    {
    case ADD:
      AddContact(&con);
      break;
    case DEL:
      DelContact(&con);
      break;
    case SEARCH:
      SearchContact(&con);
      break;
    case MODIFY:
      ModifyContact(&con);
      break;
    case SHOW:
      ShowContact(&con);
      break;
    case SORT:
      SortContact(&con);
      break;
    case EXIT:
      printf("退出通讯录\n");
      break;
    default:
      printf("选择错误,重新选择\n");
      break;
    }
  } while (input);
}
int main()
{
  test();
  return 0;
}


相关文章
|
26天前
|
存储 安全 数据管理
C语言之考勤模拟系统平台(千行代码)
C语言之考勤模拟系统平台(千行代码)
48 4
|
16天前
|
存储 算法 程序员
C 语言递归算法:以简洁代码驾驭复杂逻辑
C语言递归算法简介:通过简洁的代码实现复杂的逻辑处理,递归函数自我调用解决分层问题,高效而优雅。适用于树形结构遍历、数学计算等领域。
|
23天前
|
存储 安全 物联网
C语言物联网开发之设备安全与代码可靠性隐患
物联网设备的C语言代码安全与可靠性至关重要。一是防范代码安全漏洞,包括缓冲区溢出和代码注入风险,通过使用安全函数和严格输入验证来预防。二是提高代码跨平台兼容性,利用`stdint.h`定义统一的数据类型,并通过硬件接口抽象与适配减少平台间的差异,确保程序稳定运行。
|
17天前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
46 1
|
2月前
|
存储 搜索推荐 C语言
深入C语言指针,使代码更加灵活(二)
深入C语言指针,使代码更加灵活(二)
|
2月前
|
存储 程序员 编译器
深入C语言指针,使代码更加灵活(一)
深入C语言指针,使代码更加灵活(一)
|
2月前
|
C语言
深入C语言指针,使代码更加灵活(三)
深入C语言指针,使代码更加灵活(三)
深入C语言指针,使代码更加灵活(三)
|
2月前
|
存储 C语言
探索C语言数据结构:利用顺序表完成通讯录的实现
本文介绍了如何使用C语言中的顺序表数据结构实现一个简单的通讯录,包括初始化、添加、删除、查找和保存联系人信息的操作,以及自定义结构体用于存储联系人详细信息。
32 2
|
3月前
|
安全 C语言
在C语言中,正确使用运算符能提升代码的可读性和效率
在C语言中,运算符的使用需要注意优先级、结合性、自增自减的形式、逻辑运算的短路特性、位运算的类型、条件运算的可读性、类型转换以及使用括号来明确运算顺序。掌握这些注意事项可以帮助编写出更安全和高效的代码。
60 4
|
2月前
|
C语言
C语言练习题代码
C语言练习题代码