❤️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;
}
相关文章
|
1月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
32 3
|
5天前
|
搜索推荐 C语言
【排序算法】快速排序升级版--三路快排详解 + 实现(c语言)
本文介绍了快速排序的升级版——三路快排。传统快速排序在处理大量相同元素时效率较低,而三路快排通过将数组分为三部分(小于、等于、大于基准值)来优化这一问题。文章详细讲解了三路快排的实现步骤,并提供了完整的代码示例。
24 4
|
1月前
|
测试技术
升级电脑内存
升级电脑内存
51 2
|
18天前
|
C语言
【c语言】动态内存管理
本文介绍了C语言中的动态内存管理,包括其必要性及相关的四个函数:`malloc`、``calloc``、`realloc`和`free`。`malloc`用于申请内存,`calloc`申请并初始化内存,`realloc`调整内存大小,`free`释放内存。文章还列举了常见的动态内存管理错误,如空指针解引用、越界访问、错误释放等,并提供了示例代码帮助理解。
29 3
|
1月前
|
编译器 程序员 C语言
深入C语言:动态内存管理魔法
深入C语言:动态内存管理魔法
|
1月前
|
固态存储 内存技术
升级电脑内存和硬盘
升级电脑内存和硬盘
44 6
|
1月前
|
存储 程序员 编译器
C语言——动态内存管理与内存操作函数
C语言——动态内存管理与内存操作函数
|
1月前
|
程序员 C语言
C语言内存函数精讲
C语言内存函数精讲
|
20天前
|
存储 C语言
【c语言】字符串函数和内存函数
本文介绍了C语言中常用的字符串函数和内存函数,包括`strlen`、`strcpy`、`strcat`、`strcmp`、`strstr`、`strncpy`、`strncat`、`strncmp`、`strtok`、`memcpy`、`memmove`和`memset`等函数的使用方法及模拟实现。文章详细讲解了每个函数的功能、参数、返回值,并提供了具体的代码示例,帮助读者更好地理解和掌握这些函数的应用。
18 0
|
29天前
|
C语言
保姆级教学 - C语言 之 动态内存管理
保姆级教学 - C语言 之 动态内存管理
17 0