❤️C语言通讯录管理系统❤️ (升级版)动态开辟内存

简介: 通讯录可以用来存储1000个人的信息,每个人的信息包括:姓名、性别、年龄、电话、住址,并且实现对内存的动态开辟。

📌一、通讯录


🌿🌿实现一个通讯录;


通讯录可以用来存储1000个人的信息,每个人的信息包括:姓名、性别、年龄、电话、住址,

并且实现对内存的动态开辟。


提供方法:


添加联系人信息

删除指定联系人信息

查找指定联系人信息

修改指定联系人信息

显示所有联系人信息

清空所有联系人

以名字排序所有联系人


📌二、菜单实现和用户交互


void menu()
{
    printf("=====================\n");
    printf(" 1. 新增联系人\n");
    printf(" 2. 删除联系人\n");
    printf(" 3. 查找联系人\n");
    printf(" 4. 修改联系人\n");
    printf(" 5. 查看所有联系人\n");
    printf(" 6. 清空所有联系人\n");
    printf(" 7. 以名字排序所有联系人\n");
    printf(" 0. 退出\n");
    printf("=====================\n");
}


📌三、主函数


⭐️⭐️1.enum选项


enum Option
{
  EXIT,
  ADD,
  DEL,
  SEARCH,
  MODIFY,
  SHOW,
  EMPTY,
  SORT,
};

enum枚举常量,里面默认对应的值是从0~7,刚好和菜单中的选项匹配起来了


⭐️⭐️2.switch判断


do
  {
    menu();
    printf("请选择:>");
    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 EMPTY:
        DestroyContact(&con);
        break;
      case EXIT:
        printf("退出通讯录\n");
        break;
      default:
        printf("选择错误\n");
        break;
    }
  } while (input);


📌四、定义联系人和通讯录


⭐️⭐️1.定义联系人结构体


#define NAME_MAX 20
#define SEX_MAX 5
#define TELE_MAX 20
#define ADDR_MAX 30
struct PeoInfo
{
  char name[NAME_MAX];
  int age;
  char sex[SEX_MAX];
  char tele[TELE_MAX];
  char addr[ADDR_MAX];
};

🌿🌿注意事项:

  1. 我使用了四个define定义的常量,这样要修改最大值的时候更方便
  2. 定义的struct结构体可以存放:姓名、性别、年龄、电话、住址


⭐️⭐️2.定义动态的通讯录结构体


#define MAX 1000
struct Contact
{
  struct PeoInfo* data;//设计data指针指向的内存初始放三个元素,不够就继续开辟内存
  int sz;//记录当前通讯录有效元素个数
  int capacity;//记录当前通讯录最大容量
};

🌿🌿注意事项:

  1. 设计data指针指向的内存初始放三个元素,不够就继续开辟内存
  2. sz记录当前通讯录有效元素个数
  3. capacity记录当前通讯录最大容量


⭐️⭐️3.定义结构体变量


struct PeoInfo con;


📌五、通讯录初始化


#include<string.h>
#include"contact.h"
void InitContact(struct Contact* pc)
{
  pc->sz = 0;
  pc->data = (struct PeoInfo*)malloc(3 * sizeof(struct PeoInfo));
  pc->capacity=3;
}

🌿🌿注意事项:

  1. 用malloc开辟了一块动态内存空间,存放三个元素,每个元素的类型都是strcut

PeoInfo


📌六、新增联系人


void AddContact(struct Contact* pc)
{
  if (pc->sz == pc->capacity)
  {
    //增加容量
    struct PeoInfo*ptr=(struct PeoInfo*)realloc(pc->data, (pc->capacity + 2) * sizeof(struct PeoInfo));
    if (ptr != NULL)
    {
      pc->data = ptr;
      pc->capacity += 2;
      printf("增容成功\n");
    }
    else
    {
      return;
    }
  }
  //录入新增人的信息
  printf("请输入名字:>");
  scanf("%s", pc->data[pc->sz].name);
  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);
  printf("添加成功!\n");
}

🌿🌿注意事项:


使用realloc函数来增容。


📌七、查找联系人


🌿🌿注意事项:

  1. 我发现无论是删除联系人、修改联系人、查看所有联系人都需要一个查找所有联系人的动作,因此我们写出一个查找所有联系人的函数
int FindContactByName(struct Contact* pc, char name[])
{
  int i = 0;
  for (i = 0; i < pc->sz; i++)
  {
    if (strcmp(pc->data[i].name, name) == 0)
    {
      return i;
    }
  }
  return -1;
}

🌿🌿注意事项:

  1. strcmp库函数来比较data数组中的名字与要查找的名字是否相同
  2. 相同则返回下标,不同则返回-1
void SearchContact(struct Contact* pc)
{
  char name[NAME_MAX] = { 0 };
  printf("请输入要查找人的名字:>");
  scanf("%s", name);
  int pos= FindContactByName(pc, name);
  if (-1 == pos)
  {
    printf("查无此人");
  }
  else
  {
    printf("%15s\t%5s\t%8s\t%15s\t%30s\n\n", "姓名", "年龄", "性别", "电话", "地址");
    printf("%15s\t%5s\t%8s\t%15s\t%30s\n",
      pc->data[pos].name,
      pc->data[pos].age,
      pc->data[pos].sex,
      pc->data[pos].tele,
      pc->data[pos].addr);
  }
}


📌八、删除联系人


void DelContact(struct Contact* pc)
{
  if (pc->sz == 0)
  {
    printf("通讯录为空,无法删除\n");
  }
  char name[NAME_MAX] = { 0 };
  printf("请输入要删除人的名字");
  scanf_s("%s", name);
  int pos=FindContactByName(pc,name);//按照名字去查找,找到了就返回下标,未找到就返回-1
  if (pos == -1)
  {
    printf("指定联系人不存在\n");
  }
  else
  {
    int j = 0;
    for (j = pos; j < pc->sz-1; j++)
    {
      pc->data[j] = pc->data[j + 1];
    }
    pc->sz--;
    printf("删除成功!\n");
  }
}


📌九、修改联系人


void ModifyContact(struct Contact* pc)
{
  char name[NAME_MAX] = { 0 };
  printf("请输入要修改人的名字:>");
  scanf("%s", name);
  int pos = FindContactByName(pc, name);
  if (-1 == pos)
  {
    printf("要修改的人不存在\n");
  }
  else
  {
    printf("请输入新的名字:>");
    scanf("%s", pc->data[pos].name);//选择放在下标为sz的data里面
    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);
  }
}


📌十、查看所有联系人


void ShowContact(struct Contact* pc)
{
  int i = 0;
  printf("%15s\t%5s\t%8s\t%15s\t%30s\n\n", "姓名", "年龄", "性别", "电话", "地址");//打印标题
  for (i = 0; i < pc->sz; i++)
  {
    printf("%15s\t%5s\t%8s\t%15s\t%30s\n",
      pc->data[i].name, 
      pc->data[i].age, 
      pc->data[i].sex, 
      pc->data[i].tele, 
      pc->data[i].addr);
  }
}


📌十一、清空所有联系人


void DestroyContact(struct Contact* pc)
{
  free(pc->data);
  pc->data = NULL;
  pc->capacity = 0;
  pc->sz = 0;
}

🌿🌿注意事项:

使用free将所有动态内存释放,并且指针都变成空指针。


📌十二、以名字排序所有联系人


void SortContact(struct Contact* pc)
{
  qsort(pc->data, pc->sz, sizeof(struct PeoInfo), CmpByname);
}

🌿🌿注意事项:

有关qsort相关的知识,请参考此篇博客


📌十三、完整代码


⭐️⭐️contact.h

#define _CRT_SECURE_NO_WARNINGS 1
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#define NAME_MAX 20
#define SEX_MAX 5
#define TELE_MAX 20
#define ADDR_MAX 30
#define MAX 100
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
struct PeoInfo
{
  char name[NAME_MAX];
  int age;
  char sex[SEX_MAX];
  char tele[TELE_MAX];
  char addr[ADDR_MAX];
};
struct Contact
{
  struct PeoInfo* data;//设计data指针指向的内存初始放三个元素,不够就继续开辟内存
  int sz;//记录当前通讯录有效元素个数
  int capacity;//记录当前通讯录最大容量
};
//初始化通讯录
void InitContact(struct Contact* pc);
//销毁通讯录
void DestroyContact(struct Contact* pc);
//增加联系人
void AddContact(struct Contact* pc);
//显示所有的联系人
void ShowContact(struct Contact* pc);
//删除指定联系人
void DelContact(struct Contact* pc);
//查找指定联系人
void SearchContact(const struct Contact* pc);
//修改指定联系人
void ModifyContact(struct Contact* pc);


⭐️⭐️test.c


#define _CRT_SECURE_NO_WARNINGS 1
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include "contact.h"
void menu()
{
  printf("=====================\n");
  printf(" 1. 新增联系人\n");
  printf(" 2. 删除联系人\n");
  printf(" 3. 查找联系人\n");
  printf(" 4. 修改联系人\n");
  printf(" 5. 查看所有联系人\n");
  printf(" 6. 清空所有联系人\n");
  printf(" 7. 以名字排序所有联系人\n");
  printf(" 0. 退出\n");
  printf("=====================\n");
  printf(" 请输入您的选择:");
}
enum Option
{
  EXIT,
  ADD,
  DEL,
  SEARCH,
  MODIFY,
  SHOW,
  EMPTY,
  SORT,
};//枚举常量,里面对应的值是从0~6,刚好和菜单匹配起来了
int main()
{
  int input = 0;
  struct Contact con;
  InitContact(&con);
  //最多可以放3个人的信息
  //空间不够可以增容
  do
  {
    menu();
    printf("请选择:>");
    scanf_s("%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 EMPTY:
      //销毁通讯录
      DestroyContact(&con);
      break;
    case EXIT:
      printf("退出通讯录\n");
      break;
    default:
      printf("选择错误\n");
      break;
    }
  } while (input);
  return 0;
}

⭐️⭐️contact.c

#define _CRT_SECURE_NO_WARNINGS 1
#include<string.h>
#include "contact.h"
void InitContact(struct Contact* pc)
{
  pc->sz = 0;
  pc->data = (struct PeoInfo*)malloc(3 * sizeof(struct PeoInfo));
}
void AddContact(struct Contact* pc)
{
  if (pc->sz == pc->capacity)
  {
    //增加容量
    struct PeoInfo*ptr=(struct PeoInfo*)realloc(pc->data, (pc->capacity + 2) * sizeof(struct PeoInfo));
    if (ptr != NULL)
    {
      pc->data = ptr;
      pc->capacity += 2;
      printf("增容成功\n");
    }
    else
    {
      return;
    }
  }
  //录入新增人的信息
  printf("请输入名字:>");
  scanf("%s", pc->data[pc->sz].name);
  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);
  printf("添加成功!\n");
}
void ShowContact(struct Contact* pc)
{
  int i = 0;
  printf("%15s\t%5s\t%8s\t%15s\t%30s\n\n", "姓名", "年龄", "性别", "电话", "地址");//打印标题
  for (i = 0; i < pc->sz; i++)
  {
    printf("%15s\t%5d\t%8s\t%15s\t%30s\n",
      pc->data[i].name,
      pc->data[i].age,
      pc->data[i].sex,
      pc->data[i].tele,
      pc->data[i].addr);
  }
}
int FindContactByName(struct Contact* pc, char name[])
{
  int i = 0;
  for (i = 0; i < pc->sz; i++)
  {
    if (strcmp(pc->data[i].name, name) == 0)
    {
      return i;
    }
  }
  return -1;
}
void DelContact(struct Contact* pc)
{
  if (pc->sz == 0)
  {
    printf("通讯录为空,无法删除\n");
    return;
  }
  char name[NAME_MAX] = { 0 };
  printf("请输入要删除人的名字:>");
  scanf("%s", name);
  //查找
  int pos = FindContactByName(pc, name);
  if (pos == -1)
  {
    printf("指定的联系人不存在\n");
  }
  else
  {
    //删除
    int j = 0;
    for (j = pos; j < pc->sz - 1; j++)
    {
      pc->data[j] = pc->data[j + 1];
    }
    pc->sz--;
    //
    printf("删除成功\n");
  }
}
void SearchContact(struct Contact* pc)
{
  char name[NAME_MAX] = { 0 };
  printf("请输入要查找人的名字:>");
  scanf("%s", name);
  int pos = FindContactByName(pc, name);
  if (-1 == pos)
  {
    printf("查无此人");
  }
  else
  {
    printf("%15s\t%5s\t%8s\t%15s\t%30s\n\n", "姓名", "年龄", "性别", "电话", "地址");
    printf("%15s\t%5d\t%8s\t%15s\t%30s\n",
      pc->data[pos].name,
      pc->data[pos].age,
      pc->data[pos].sex,
      pc->data[pos].tele,
      pc->data[pos].addr);
  }
}
void ModifyContact(struct Contact* pc)
{
  char name[NAME_MAX] = { 0 };
  printf("请输入要修改人的名字:>");
  scanf("%s", name);
  int pos = FindContactByName(pc, name);
  if (-1 == pos)
  {
    printf("要修改的人不存在\n");
  }
  else
  {
    printf("请输入新的名字:>");
    scanf("%s", pc->data[pos].name);//选择放在下标为sz的data里面
    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);
  }
}
void DestroyContact(struct Contact* pc)
{
  free(pc->data);
  pc->data = NULL;
  pc->capacity = 0;
  pc->sz = 0;
}
相关文章
|
6月前
|
缓存 监控 Linux
CentOS系统如何查看当前内存容量。
以上方法都不需要特殊软件或者复杂配置即可执行,在CentOS或其他Linux发行版中都适合运行,并且它们各自透露出不同角度对待问题解答方式:从简单快速到深入详尽;从用户态到核心态;从操作层数到硬件层数;满足不同用户需求与偏好。
472 8
|
7月前
|
存储 缓存 监控
手动清除Ubuntu系统中的内存缓存的步骤
此外,只有系统管理员或具有适当权限的用户才能执行这些命令,因为这涉及到系统级的操作。普通用户尝试执行这些操作会因权限不足而失败。
1387 22
|
7月前
|
安全 C语言
C语言中的字符、字符串及内存操作函数详细讲解
通过这些函数的正确使用,可以有效管理字符串和内存操作,它们是C语言编程中不可或缺的工具。
380 15
|
11月前
|
监控 Linux Python
Linux系统资源管理:多角度查看内存使用情况。
要知道,透过内存管理的窗口,我们可以洞察到Linux系统运行的真实身姿,如同解剖学家透过微观镜,洞察生命的奥秘。记住,不要惧怕那些高深的命令和参数,他们只是你掌握系统"魔法棒"的钥匙,熟练掌握后,你就可以骄傲地说:Linux,我来了!
395 27
|
IDE 编译器 开发工具
【C语言】全面系统讲解 `#pragma` 指令:从基本用法到高级应用
在本文中,我们系统地讲解了常见的 `#pragma` 指令,包括其基本用法、编译器支持情况、示例代码以及与传统方法的对比。`#pragma` 指令是一个强大的工具,可以帮助开发者精细控制编译器的行为,优化代码性能,避免错误,并确保跨平台兼容性。然而,使用这些指令时需要特别注意编译器的支持情况,因为并非所有的 `#pragma` 指令都能在所有编译器中得到支持。
1462 41
【C语言】全面系统讲解 `#pragma` 指令:从基本用法到高级应用
|
存储 缓存 算法
JDK11升级后竟让内存利用率飙升到90%以上?
本文记录了作者升级到JDK11后,使用G1GC导致内存利用率飙升至90%以上的问题及其解决方案。
|
存储 编译器 程序员
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
在C语言中,内存布局是程序运行时非常重要的概念。内存布局直接影响程序的性能、稳定性和安全性。理解C程序的内存布局,有助于编写更高效和可靠的代码。本文将详细介绍C程序的内存布局,包括代码段、数据段、堆、栈等部分,并提供相关的示例和应用。
611 5
【C语言】内存布局大揭秘 ! -《堆、栈和你从未听说过的内存角落》
|
12月前
|
存储 缓存 算法
JDK11升级后竟让内存利用率飙升到90%以上?
JDK11升级后竟让内存利用率飙升到90%以上?
|
存储 编译器 C语言
【C语言】C语言的变量和声明系统性讲解
在C语言中,声明和定义是两个关键概念,分别用于告知编译器变量或函数的存在(声明)和实际创建及分配内存(定义)。声明可以多次出现,而定义只能有一次。声明通常位于头文件中,定义则在源文件中。通过合理组织头文件和源文件,可以提高代码的模块化和可维护性。示例包括全局变量、局部变量、函数、结构体、联合体、数组、字符串、枚举和指针的声明与定义。
578 12
|
监控 关系型数据库 MySQL
【01】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-硬件设备实时监控系统运营版发布-本产品基于企业级开源项目Zabbix深度二开-分步骤实现预计10篇合集-自营版
【01】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-硬件设备实时监控系统运营版发布-本产品基于企业级开源项目Zabbix深度二开-分步骤实现预计10篇合集-自营版
449 0