C语言——通讯录实现(增删查改+动态管理+文件操作)

简介: C语言——通讯录实现(增删查改+动态管理+文件操作)

C语言——通讯录

前言:

要实现通讯录的增删查改,动态管理,文件操作,你首先需要掌握以下内容:

头文件定义

所需头文件:

#include<stdio.h> //用于标准输入输出
#include<stdlib.h>  //用于动态内存开辟
#include<stdbool.h> //用于返回bool类型
#include<assert.h>  //用于使用断言
#include<string.h>  //用于字符串操作函数

为了方便后续的修改,我们定义几个全局变量来确定每个成员信息的长度大小

#define Max_Number_Init 3 //初始可容纳成员个数
#define Max_Name 20   //成员姓名的最大长度
#define Max_Sex 5   //成员性别的最大长度
#define Max_PhoneNumber 15  //成员电话的最大长度
#define Max_Address 12  //成员地址的最大长度

定义成员信息的结构体:

typedef struct PeopleInformation
{
  char name[Max_Name];  //姓名
  int age;  //年龄
  char sex[Max_Sex];  //性别
  char PhoneNumber[Max_PhoneNumber];  //电话
  char address[Max_Address];  //地址
}PeoInfo;

定义通讯录的结构体:

typedef struct Contact
{
  PeoInfo* nums;  //储存成员信息的数组
  int size; //记录通讯录已经容纳的成员个数
  int capacity; //记录通讯录的最大容量
}Con;

实现主函数(int main())

打印选择菜单

void meau()
{
  printf("*******************************\n");
  printf("****1->Add         2->Del******\n");
  printf("****3->Search      4->Modify***\n");
  printf("****5->Show        6->Claer****\n");
  printf("****0->Exit          **********\n");
}

为了提高代码的可读性,我们可以定义一个枚举类型,来分别代表这些选项:

enum Choice
{
  Exit, //0
  Add,  //1
  Del,  //2
  Search, //3
  Modify, //4
  Show, //5
  Clear //6
};

int main()

int main()
{
  int input = 0;
    //为通讯录申请内存
  Con* con = (Con*)malloc(sizeof(Con));
  ContactInit(con); //初始化通讯录
  do
  {
    meau();
    printf("请输入需要的选项:");
    scanf("%d", &input);
    switch (input)
    {
    case Add:
      ContactPush(con); //增
      break;
    case Del:
      ContactPop(con);  //删
      break;
    case Search:
      ContactSearch(con); //查
      break;
    case Modify:
      ContactModify(con); //改
      break;
    case Show:
      ContactShow(con); //展示通讯录
      break;
    case Clear:
      ContactClear(con);  //清空通讯录
      break;
    case Exit:
      ContactDataInput(con);  //退出通讯录
      printf("退出通讯录\n");
      break;
    default:
      printf("输入错误,请重新输入\n");
      break;
    }
  } while (input);
    //释放动态内存
  if(con->nums)
    free(con->nums);
  free(con);
  return 0;
}

ContactClear()

void ContactClear(Con* con)

清空通讯录,就是将保存成员信息的数组con->nums释放,同时置空,并且将size置0

void ContactClear(Con* con)
{
  free(con->nums);
  con->nums = NULL;
  con->size = 0;
}

ContactInit()

void ContactInit(Con* con);

通讯录的初始化就是为储存成员信息的数组申请内存,确定最大容量以及将size置0

void ContactInit(Con* con)
{
  con->capacity = Max_Number_Init;
  con->nums = (PeoInfo*)malloc(sizeof(PeoInfo) * con->capacity);
  con->size = 0;
}

ContactPush()

void ContactPush(Con* con)

在加入成员之前,我们需要检查是否还有空间可以储存,如果已经满了,就要增容

我们可以单独写一个函数来进行检查和增容:

ContactCheak()

bool ContactFull(const Con* con)
{
  assert(con);
  return con->size == con->capacity;
}
void ContactCheak(Con* con)
{
  assert(con);
  if (ContactFull(con))
  {
    con->capacity += 2; //如果满了,每次就将最大容量加2
        //增容
    PeoInfo* temp = (PeoInfo*)realloc(con->nums, con->capacity * sizeof(PeoInfo));
    if (temp == NULL)
    {
      perror("realloc");
      exit(0);
    }
    con->nums = temp;
  }
}

ContactPush()

void ContactPush(Con* con)
{
  assert(con);
  ContactCheak(con);
    //如果通讯录之前被清空过,要想重新使用,就先要为nums申请内存
  if (NULL == con->nums)
  {
    con->nums = (PeoInfo*)malloc(sizeof(PeoInfo) * Max_Number_Init);
    if (NULL == con->nums)
    {
      perror("malloc");
      exit(1);
    }
  }
  printf("请输入添加的联系人姓名:");
  scanf("%s", (con->nums + con->size)->name);
  printf("请输入该联系人的年龄:");
  scanf("%d", &(con->nums + con->size)->age);
  printf("请输入该联系人的性别:");
  scanf("%s", (con->nums + con->size)->sex);
  printf("请输入该联系人的电话:");
  scanf("%s", (con->nums + con->size)->PhoneNumber);
  printf("请输入该联系人的地址:");
  scanf("%s", (con->nums + con->size)->address);
  con->size++;
  printf("成功添加联系人\n");
}

ContactPop()

void ContactPop(Con* con)

这里通过姓名来删除联系人

  • 最开始,要判断通讯录是否为空,如果为空,就不能删除
  • 在删除之前,要现在通讯录中查找是否有这个联系人,如果没有,就不要删除(我们用一个函数ContactFind来实现查找成员,并返回成员下标)
  • 为了保证删除效率,我们找到需要删除的联系人的下标后,将其和最后一个成员交换位置,再利用size--删除最后一个成员即可

ContactEmpty()

bool ContactEmpty(const Con* con)
{
  assert(con);
  return con->size == 0;
}

Swap()

void Swap(PeoInfo* peo1, PeoInfo* peo2)
{
  PeoInfo temp = *peo1;
  *peo1 = *peo2;
  *peo2 = temp;
}

ContactFind()

int ContactFind(const Con* con, const char* str)
{
  for (int i = 0; i < con->size; i++)
  {
    if (strcmp(con->nums[i].name, str) == 0)
      return i;
  }
  return -1;
}

ContactPop()

void ContactPop(Con* con)
{
  assert(con);  //判断指针有效性
  assert(!ContactEmpty(con)); //通讯录不为空
  char name[20];
  printf("请输入被查找人的姓名:");
  scanf("%s", name);
  int index = ContactFind(con, name);
  if (index != -1)
  {
        //打印将被删除的成员信息
    printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址");
    printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n",
      (con->nums + index)->name,
      (con->nums + index)->age,
      (con->nums + index)->sex,
      (con->nums + index)->PhoneNumber,
      (con->nums + index)->address);
        //清空缓冲区
    while (getchar() != '\n');
    printf("确定删除(Y|N): ");
    char ch = getchar();
    if (ch == 'Y')
    {
      Swap(&con->nums[index], &con->nums[con->size - 1]);
      con->size--;
      printf("删除成功\n");
    }
    else
    {
      printf("删除失败\n");
    }
  }
  else
  {
    printf("未找到该联系人\n");
  }
}

ContactSearch()

void ContactSearch(const Con* con)

这里统一用姓名来查找联系人,找到便打印联系人信息

void ContactSearch(const Con* con)
{
  assert(con);
  char name[20];
  printf("请输入被查找人的姓名:");
  scanf("%s", name);
  int index = ContactFind(con, name);
  if (index != -1)
  {
    printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址");
    printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n",
      (con->nums + index)->name,
      (con->nums + index)->age,
      (con->nums + index)->sex,
      (con->nums + index)->PhoneNumber,
      (con->nums + index)->address);
  }
  else
  {
    printf("未找到该联系人\n");
  }
}

ContactModify()

void ContactModify(Con* con)

这里统一用姓名查找联系人,然后再进行修改

void ContactModify(Con* con)
{
  assert(con);
  char name[20];
  printf("请输入被查找人的姓名:");
  scanf("%s", name);
  int index = ContactFind(con, name);
  if (index != -1)
  {
    char ch;
        //展示联系人姓名
    printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址");
    printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n\n",
      (con->nums + index)->name,
      (con->nums + index)->age,
      (con->nums + index)->sex,
      (con->nums + index)->PhoneNumber,
      (con->nums + index)->address);
    printf("是否修改年龄(Y|N): ");
    while (getchar() != '\n');
    ch = getchar();
    if (ch == 'Y')
    {
      printf("请输入新的联系人年龄:");
      scanf("%d", &con->nums[index].age);
    }
    printf("是否修改性别(Y|N): ");
    while (getchar() != '\n');
    ch = getchar();
    if (ch == 'Y')
    {
      printf("请输入新的联系人性别:");
      scanf("%s", con->nums[index].sex);
    }
    printf("是否修改电话(Y|N): ");
    while (getchar() != '\n');
    ch = getchar();
    if (ch == 'Y')
    {
      printf("请输入新的联系人电话:");
      scanf("%s", con->nums[index].PhoneNumber);
    }
    printf("是否修改住址(Y|N): ");
    while (getchar() != '\n');
    ch = getchar();
    if (ch == 'Y')
    {
      printf("请输入新的联系人地址:");
      scanf("%s", con->nums[index].address);
    }
  }
  else
  {
    printf("未找到该联系人\n");
  }
}

ContactShow()

  • 展示联系人信息
  • 这里我们再加一个功能,即展示的时候先将所有联系人按姓名排序,在进行打印

ContactSort()

int compare_by_name(const void* data1, const void* data2)
{
  return strcmp(((PeoInfo*)data1)->name, ((PeoInfo*)data2)->name);
}
void ContactSort(Con* con)
{
    //用qsort进行排序
  qsort(con->nums, con->size, sizeof(PeoInfo), compare_by_name);
}

ContactShow()

void ContactShow(const Con* con)
{
  assert(con);
  ContactSort(con);
  printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n","姓名","年龄","性别","电话","地址");
  for (int i = 0; i < con->size; i++)
  {
    printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n",
      (con->nums + i)->name,
      (con->nums + i)->age,
      (con->nums + i)->sex,
      (con->nums + i)->PhoneNumber,
      (con->nums + i)->address);
  }
}

文件操作

要实现文件操作,我们再加入两个函数就可以了

  • 一个是打算退出通讯录的时候,我们将所有联系人的信息都存入文件(ContactDataInput()
  • 一个是打开通讯录,并进行初始化的时候,我们先将文件中的联系人信息存入通讯录(ContactDataCopy()

ContactDataInput()

void ContactDataInput(Con* con)
{
  assert(con);
  FILE* fp = fopen("data.txt", "w");
  if (NULL == fp)
  {
    perror("fopen");
    exit(1);
  }
  for (int i = 0; i < con->size; i++)
    fprintf(fp, "%s %d %s %s %s %c", con->nums[i].name, con->nums[i].age, con->nums[i].sex, con->nums[i].PhoneNumber, con->nums[i].address, '\n');
  fclose(fp);
  fp = NULL;
}

ContactDataCopy()

void ContactDataCopy(Con* con)
{
  assert(con);
  FILE* fp = fopen("data.txt", "r");
  if (NULL == fp)
  {
    perror("fopen");
    exit(1);
  }
  PeoInfo temp = {0};
  while (fscanf(fp, "%s %d %s %s %s", temp.name, &temp.age, temp.sex, temp.PhoneNumber, temp.address)!=EOF)
  {
        //插入之前先检查容量
    ContactCheak(con);
    con->nums[con->size] = temp;
    con->size++;
  }
  fclose(fp);
  fp = NULL;
}

实现代码:

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#include<assert.h>
#include<string.h>
#define Max_Number_Init 3
#define Max_Name 20
#define Max_Sex 5
#define Max_PhoneNumber 15
#define Max_Address 12
typedef struct PeopleInformation
{
  char name[Max_Name];
  int age;
  char sex[Max_Sex];
  char PhoneNumber[Max_PhoneNumber];
  char address[Max_Address];
}PeoInfo;
typedef struct Contact
{
  PeoInfo* nums;
  int size;
  int capacity;
}Con;
void ContactInit(Con* con);
bool ContactEmpty(const Con* con);
bool ContactFull(const Con* con);
void ContactPush(Con* con);
void ContactPop(Con* con);
void ContactSearch(const Con* con);
void ContactModify(Con* con);
void ContactShow(const Con* con);
void ContactClear(Con* con);
void ContactDataInput(Con* con);
void ContactDataCopy(Con* con);
void ContactCheak(Con* con)
{
  assert(con);
  if (ContactFull(con))
  {
    con->capacity += 2;
    PeoInfo* temp = (PeoInfo*)realloc(con->nums, con->capacity * sizeof(PeoInfo));
    if (temp == NULL)
    {
      perror("realloc");
      exit(0);
    }
    con->nums = temp;
  }
}
void ContactDataCopy(Con* con)
{
  assert(con);
  FILE* fp = fopen("data.txt", "r");
  if (NULL == fp)
  {
    perror("fopen");
    exit(1);
  }
  PeoInfo temp = {0};
  while (fscanf(fp, "%s %d %s %s %s", temp.name, &temp.age, temp.sex, temp.PhoneNumber, temp.address)!=EOF)
  {
    ContactCheak(con);
    con->nums[con->size] = temp;
    con->size++;
  }
  fclose(fp);
  fp = NULL;
}
int compare_by_name(const void* data1, const void* data2)
{
  return strcmp(((PeoInfo*)data1)->name, ((PeoInfo*)data2)->name);
}
void ContactSort(Con* con)
{
  qsort(con->nums, con->size, sizeof(PeoInfo), compare_by_name);
}
int ContactFind(const Con* con, const char* str)
{
  for (int i = 0; i < con->size; i++)
  {
    if (strcmp(con->nums[i].name, str) == 0)
      return i;
  }
  return -1;
}
void ContactInit(Con* con)
{
  con->capacity = Max_Number_Init;
  con->nums = (PeoInfo*)malloc(sizeof(PeoInfo) * con->capacity);
  con->size = 0;
  ContactDataCopy(con);
}
bool ContactEmpty(const Con* con)
{
  assert(con);
  return con->size == 0;
}
bool ContactFull(const Con* con)
{
  assert(con);
  return con->size == con->capacity;
}
void ContactShow(const Con* con)
{
  assert(con);
  ContactSort(con);
  printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n","姓名","年龄","性别","电话","地址");
  for (int i = 0; i < con->size; i++)
  {
    printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n",
      (con->nums + i)->name,
      (con->nums + i)->age,
      (con->nums + i)->sex,
      (con->nums + i)->PhoneNumber,
      (con->nums + i)->address);
  }
}
void ContactPush(Con* con)
{
  assert(con);
  ContactCheak(con);
  if (NULL == con->nums)
  {
    con->nums = (PeoInfo*)malloc(sizeof(PeoInfo) * Max_Number_Init);
    if (NULL == con->nums)
    {
      perror("malloc");
      exit(1);
    }
  }
  printf("请输入添加的联系人姓名:");
  scanf("%s", (con->nums + con->size)->name);
  printf("请输入该联系人的年龄:");
  scanf("%d", &(con->nums + con->size)->age);
  printf("请输入该联系人的性别:");
  scanf("%s", (con->nums + con->size)->sex);
  printf("请输入该联系人的电话:");
  scanf("%s", (con->nums + con->size)->PhoneNumber);
  printf("请输入该联系人的地址:");
  scanf("%s", (con->nums + con->size)->address);
  con->size++;
  printf("成功添加联系人\n");
}
void Swap(PeoInfo* peo1, PeoInfo* peo2)
{
  PeoInfo temp = *peo1;
  *peo1 = *peo2;
  *peo2 = temp;
}
void ContactPop(Con* con)
{
  assert(con);
  assert(!ContactEmpty(con));
  char name[20];
  printf("请输入被查找人的姓名:");
  scanf("%s", name);
  int index = ContactFind(con, name);
  if (index != -1)
  {
    printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址");
    printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n",
      (con->nums + index)->name,
      (con->nums + index)->age,
      (con->nums + index)->sex,
      (con->nums + index)->PhoneNumber,
      (con->nums + index)->address);
    while (getchar() != '\n');
    printf("确定删除(Y|N): ");
    char ch = getchar();
    if (ch == 'Y')
    {
      Swap(&con->nums[index], &con->nums[con->size - 1]);
      con->size--;
      printf("删除成功\n");
    }
    else
    {
      printf("删除失败\n");
    }
  }
  else
  {
    printf("未找到该联系人\n");
  }
}
void ContactSearch(const Con* con)
{
  assert(con);
  char name[20];
  printf("请输入被查找人的姓名:");
  scanf("%s", name);
  int index = ContactFind(con, name);
  if (index != -1)
  {
    printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址");
    printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n",
      (con->nums + index)->name,
      (con->nums + index)->age,
      (con->nums + index)->sex,
      (con->nums + index)->PhoneNumber,
      (con->nums + index)->address);
  }
  else
  {
    printf("未找到该联系人\n");
  }
}
void ContactModify(Con* con)
{
  assert(con);
  char name[20];
  printf("请输入被查找人的姓名:");
  scanf("%s", name);
  int index = ContactFind(con, name);
  if (index != -1)
  {
    char ch;
    printf("%-20s\t%-4s\t%-5s\t%-15s\t%-12s\n", "姓名", "年龄", "性别", "电话", "地址");
    printf("%-20s\t%-4d\t%-5s\t%-15s\t%-12s\n\n",
      (con->nums + index)->name,
      (con->nums + index)->age,
      (con->nums + index)->sex,
      (con->nums + index)->PhoneNumber,
      (con->nums + index)->address);
    printf("是否修改年龄(Y|N): ");
    while (getchar() != '\n');
    ch = getchar();
    if (ch == 'Y')
    {
      printf("请输入新的联系人年龄:");
      scanf("%d", &con->nums[index].age);
    }
    printf("是否修改性别(Y|N): ");
    while (getchar() != '\n');
    ch = getchar();
    if (ch == 'Y')
    {
      printf("请输入新的联系人性别:");
      scanf("%s", con->nums[index].sex);
    }
    printf("是否修改电话(Y|N): ");
    while (getchar() != '\n');
    ch = getchar();
    if (ch == 'Y')
    {
      printf("请输入新的联系人电话:");
      scanf("%s", con->nums[index].PhoneNumber);
    }
    printf("是否修改住址(Y|N): ");
    while (getchar() != '\n');
    ch = getchar();
    if (ch == 'Y')
    {
      printf("请输入新的联系人地址:");
      scanf("%s", con->nums[index].address);
    }
  }
  else
  {
    printf("未找到该联系人\n");
  }
}
void ContactClear(Con* con)
{
  free(con->nums);
  con->nums = NULL;
  con->size = 0;
}
void ContactDataInput(Con* con)
{
  assert(con);
  FILE* fp = fopen("data.txt", "w");
  if (NULL == fp)
  {
    perror("fopen");
    exit(1);
  }
  for (int i = 0; i < con->size; i++)
    fprintf(fp, "%s %d %s %s %s %c", con->nums[i].name, con->nums[i].age, con->nums[i].sex, con->nums[i].PhoneNumber, con->nums[i].address, '\n');
  fclose(fp);
  fp = NULL;
}
void meau()
{
  printf("*******************************\n");
  printf("****1->Add         2->Del******\n");
  printf("****3->Search      4->Modify***\n");
  printf("****5->Show        6->Claer****\n");
  printf("****0->Exit          **********\n");
}
enum Choice
{
  Exit,
  Add,
  Del,
  Search,
  Modify,
  Show,
  Clear
};
int main()
{
  int input = 0;
  Con* con = (Con*)malloc(sizeof(Con));
  ContactInit(con);
  do
  {
    meau();
    printf("请输入需要的选项:");
    scanf("%d", &input);
    switch (input)
    {
    case Add:
      ContactPush(con);
      break;
    case Del:
      ContactPop(con);
      break;
    case Search:
      ContactSearch(con);
      break;
    case Modify:
      ContactModify(con);
      break;
    case Show:
      ContactShow(con);
      break;
    case Clear:
      ContactClear(con);
      break;
    case Exit:
      ContactDataInput(con);
      printf("退出通讯录\n");
      break;
    default:
      printf("输入错误,请重新输入\n");
      break;
    }
  } while (input);
  if(con->nums)
    free(con->nums);
  free(con);
  return 0;
}
相关文章
|
1月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
51 9
|
1月前
|
存储 数据管理 C语言
C 语言中的文件操作:数据持久化的关键桥梁
C语言中的文件操作是实现数据持久化的重要手段,通过 fopen、fclose、fread、fwrite 等函数,可以实现对文件的创建、读写和关闭,构建程序与外部数据存储之间的桥梁。
|
2月前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
129 3
|
2月前
|
存储 C语言
【c语言】玩转文件操作
本文介绍了C语言中文件操作的基础知识,包括文件的打开和关闭、文件的顺序读写、文件的随机读写以及文件读取结束的判定。详细讲解了`fopen`、`fclose`、`fseek`、`ftell`、`rewind`等函数的使用方法,并通过示例代码展示了如何进行文件的读写操作。最后,还介绍了如何判断文件读取结束的原因,帮助读者更好地理解和应用文件操作技术。
52 2
|
3月前
|
存储 C语言
C语言文件操作(2)
【10月更文挑战第2天】
|
3月前
|
存储 C语言
探索C语言数据结构:利用顺序表完成通讯录的实现
本文介绍了如何使用C语言中的顺序表数据结构实现一个简单的通讯录,包括初始化、添加、删除、查找和保存联系人信息的操作,以及自定义结构体用于存储联系人详细信息。
42 2
|
3月前
|
程序员 编译器 C语言
C语言底层知识------文件操作
本文详细介绍了文件操作的基本概念,包括文件的分类(程序文件和数据文件,其中着重于数据文件的文本文件和二进制文件),流的概念及其在C程序中的应用,以及标准输入输出流stdin、stdout和stderr的作用。作者通过示例展示了如何使用fopen、fclose和常见的读写函数如fgetc、fputc和fgets进行文件操作。
35 2
|
3月前
|
存储 缓存 编译器
文件操作——C语言
文件操作——C语言
|
3月前
|
存储 C语言
简述C语言文件操作
简述C语言文件操作
13 0
|
3月前
|
存储 文件存储 C语言
深入C语言:文件操作实现局外影响程序
深入C语言:文件操作实现局外影响程序