【C语言】通讯录实现

简介: 通讯录功能

通讯录功能

添加联系人信息(名字,性别,年龄,电话号码,家庭住址)

输出指定联系人信息

查找指定联系人信息

修改指定联系人信息

打印所有联系人信息

对所有联系人(通过名字)排序

保存当前通讯录内容(以二进制形式保存到当前文件夹中的contactinformation这个数据文件里)

文件分装

20210224150126978.png


对应文件的代码

contact.h

//声明C库函数的头文件
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<errno.h>
//定义相关量的宏
//struct PeoInfo数组成员的默认大小 
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30
//初始化时,通讯录的默认容量大小
#define DEFAULT_SIZE 3
//枚举,作为case语句的执行常量表达式
enum Option
{
  EXIT,//0
  ADD,//1
  DEL,//2
  SEARCH,//3
  MODIFY,//4
  SHOW,//5
  SORT,//6
  SAVE//7
};
//声明成员信息的结构体
struct PeoInfo
{
  char name[MAX_NAME];
  char sex[MAX_SEX];
  int age;
  char tele[MAX_TELE];
  char addr[MAX_ADDR];
};
//声明通讯录信息的结构体
struct Contact
{
  struct PeoInfo* data;//存放所有成员的数据
  int size;//记录通讯录当前有多少人
  int capacity;//记录通讯录当前容量大小
};
//函数声明
void InitContact(struct Contact* p);
void AddContact(struct Contact* p);
void ShowContact(struct Contact* p);
void SearchContact(struct Contact* p);
void DelContact(struct Contact* p);
void ModifyContact(struct Contact* p);
void SortContact(struct Contact* p);
void DistroyContact(struct Contact* p);
void SaveContact(struct Contact* p);


contact.c

#include"contact.h"
//检查通讯录的容量(内存空间)是否足够
//不够就在开辟两个成员空间
//够了就什么都不做
static void CheckCapacity(struct Contact* p)
{
  //判断当前通讯录容量是否满了
  if (p->size == p->capacity)//当前通讯录成员人数是否等于最大容量
  {
  //开辟空间
  struct PeoInfo* tmp = (struct PeoInfo*)realloc(p->data, (2 + p->capacity) * sizeof(struct PeoInfo));
  if (tmp != NULL)//开辟成功
  {
    p->data = tmp;
    p->capacity += 2;
  }
  else//开辟失败
  {
    printf("CheckCapacity::%s\n", strerror(errno));
    return;
  }
  }
}
//上传上次写在文件里的成员信息
static void LoadContact(struct Contact* p)
{
  //以只读的方式打开文件
  FILE* pf = fopen("contactinformation", "rb");
  if (pf == NULL)//检查是否打开
  {
  printf("LoadContact::%s\n", strerror(errno));
  }
  struct PeoInfo tmp = { 0 };//创建一个临时的结构体变量
  //开始逐条读取pf的(成员信息)内容放到中间变量tmp里
  while (fread(&tmp, sizeof(struct PeoInfo), 1, pf))
  {
  CheckCapacity(p);//检查当前的结构体容量是否可以放得下所有之前的成员信息
  *(p->data + p->size) = tmp;
  p->size++;//每放一个成员,记录成员人数的size就加一
  }
  //文件读取完毕,关闭文件
  fclose(pf);
  pf = NULL;
}
//初始化通讯录信息
void InitContact(struct Contact* p)
{
  //为通讯录DEFAULT_SIZE个成员的空间
  p->data = (struct PeoInfo*)malloc(DEFAULT_SIZE * sizeof(struct PeoInfo));
  if (p->data == NULL)//检查是否开辟成功
  {
  printf("InitContact::%s\n", strerror(errno));
  return;
  }
  p->size = 0;
  p->capacity= DEFAULT_SIZE;
  LoadContact(p);//初始化之后把以前写在文件里的成员信息上次到这次程序的通讯录里
}
//增加成员信息
void AddContact(struct Contact* p)
{
  void CheckCapacity(p);//检查当前通讯录容量是否足够,不够的话就开辟空间,够了就什么都不干
  printf("请输入名字:>");
  scanf("%s",(*((p->data)+(p->size))).name);
  printf("请输入性别:>");
  scanf("%s", (*((p->data) + (p->size))).sex);
  printf("请输入年龄:>");
  scanf("%d", &(*((p->data) + (p->size))).age);
  printf("请输入电话号码:>");
  scanf("%s", (*((p->data) + (p->size))).tele);
  printf("请输入住址:>");
  scanf("%s", (*((p->data) + (p->size))).addr);
  printf("添加成功\n");
  p->size++;//添加完成之后记录当前通讯录成员个数的size变量加一
}
//查看通讯录中所有成员信息
void ShowContact(struct Contact* p)
{
  //先检查当前通讯录是否有成员信息
  if (p->size == 0)//没有成员
  {
  printf("通讯录内容为空\n");
  }
  else//有成员,依次打印每个成员的信息
  {
  printf("%-20s\t%-4s\t%-5s\t%-12s\t%-20s\n", "名字", "性别", "年龄", "电话", "住址");
  int i = 0;
  for (i = 0; i < p->size; i++)
  {
    printf("%-20s\t%-4s\t%-5d\t%-12s\t%-20s\n",
    (*((p->data) + i)).name,
    (*((p->data) + i)).sex,
    (*((p->data) + i)).age,
    (*((p->data) + i)).tele,
    (*((p->data) + i)).addr);
  }
  }
}
//实现查找成员名字功能的函数
//找到了就返回成员的下标
//找不到就返回-1
static int Search_By_Name(struct Contact* p,char* pc)
{
  int i = 0;
  for (i = 0; i < p->size; i++)
  {
  if (strcmp((*((p->data) + i)).name, pc) == 0)
  {
    return i;
  }
  }
  return -1;
}
//查找指定成员
void SearchContact(struct Contact* p)
{
  char name[MAX_NAME] = { 0 };
  printf("请输入要查找人的姓名:>");
  scanf("%s", name);
  //判断该成员是否存在
  int flag = Search_By_Name(p, name);
  if (flag == -1)//不存在
  {
  printf("要查找人的信息不存在\n");
  }
  else//存在,打印该成员的信息
  {
  printf("%-20s\t%-4s\t%-5s\t%-12s\t%-20s\n", "名字", "性别", "年龄", "电话", "住址");
  printf("%-20s\t%-4s\t%-5d\t%-12s\t%-20s\n",
    (*((p->data) + flag)).name,
    (*((p->data) + flag)).sex,
    (*((p->data) + flag)).age,
    (*((p->data) + flag)).tele,
    (*((p->data) + flag)).addr);
  }
}
//输出通讯录中指定成员信息
void DelContact(struct Contact* p)
{
  char name[MAX_NAME] = { 0 };
  printf("请输入要删除人的名字:>");
  scanf("%s", name);
  //判断该成员是否存在
  int flag = Search_By_Name(p, name);
  if (flag == -1)//不存在
  {
  printf("要删除的人不存在\n");
  }
  else//存在,把后面成员的位置往前挪一位
  {
  int i = 0;
  for (i = flag; i < p->size-1; i++)
  {
    *(p->data + i) = *(p->data + i + 1);
  }
  printf("删除成功\n");
  p->size--;//令记录当前通讯录成员个数的size减一
  }
}
//修改通讯录中指定成员信息
void ModifyContact(struct Contact* p)
{
  char name[MAX_NAME] = { 0 };
  printf("请输入要修改人的名字:>");
  scanf("%s", name);
  //判断该成员是否存在
  int flag=Search_By_Name(p, name);
  if (flag == -1)//不存在
  {
  printf("要修改人的信息不存在\n");
  }
  else//存在,通过访问结构体成员直接修改信息
  {
  printf("请输入名字:>");
  scanf("%s", (*((p->data) + flag)).name);
  printf("请输入性别:>");
  scanf("%s", (*((p->data) + flag)).sex);
  printf("请输入年龄:>");
  scanf("%d", &(*((p->data) + flag)).age);
  printf("请输入电话号码:>");
  scanf("%s", (*((p->data) + flag)).tele);
  printf("请输入住址:>");
  scanf("%s", (*((p->data) + flag)).addr);
  printf("修改成功\n");
  }
}
//分隔行内都是实现(通过名字)排序功能的相关函数
//-------------------------------------------------------------------------------------------------------------------------------------------------------------
//自定义的比较函数(通过名字来比较两个元素的大小)
//比较两个元素
//p1>p2返回大于零的数字
//p1=p2返回0
//p1<p2返回小于零的数字
static int Cmp_Struct(void* p1, void* p2)
{
  return strcmp(((struct PeoInfo*)p1)->name , ((struct PeoInfo*)p2)->name);
}
//交换两个元素的内容(width,是一个元素所占内存空间的大小,单位是字节)
static void Swap(void* p1, void* p2, int width)
{
  //一个字节一个字节的交换
  while (width)
  {
  char tmp = *(char*)p1;
  *(char*)p1 = *(char*)p2;
  *(char*)p2 = tmp;
  ++(char*)p1;
  ++(char*)p2;
  width--;
  }
}
//通用的冒泡排序(什么类型的数据都可以排)
static void My_Bubble_Sort(void* p, int sz, int width, int(*cmp)(void*, void*))
{
  int i = 0;
  int j = 0;
  int flag = 1;//flag为1是是已经排好序,为0是是还没排好序,先默认它为1
  //冒泡排序
  for (i = 0; i < sz - 1; i++)//sz个元素,一共要比较sz-1趟,每一趟排序好一个元素
  {
  for (j = 0; j < sz - 1 - i; j++)//每趟要两两比较当前未排好序的元素个数-1次
  {
    if (cmp((char*)p + (j*width), (char*)p + ((j + 1)*width)) > 0)//判断是否要交换两个元素(既判断前一个元素的名字是否大于后一个元素的名字)
    {
    Swap((char*)p + (j*width), (char*)p + ((j + 1)*width), width);//交换两个成员的位置
    flag = 0;//已经判断这次是无序了,所以flag=0
    }
  }
  if (flag == 1)//若一趟两两比较下来flag仍然=1,则数组肯定有序了,跳出冒泡排序
  {
    break;
  }
  }
}
void SortContact(struct Contact* p)
{
  //排序之前的传参
  My_Bubble_Sort(p->data,p->size,sizeof(p->data[0]),Cmp_Struct);
  //p->data:成员信息数组的首元素地址
  //p->size:当前成员个数
  //sizeof(p->data[0]):一个成员所占内存空间的大小,单位是字节
  //Cmp_Struct:自定义的比较两个元素功能的函数
  printf("排序完成\n");
}
//-------------------------------------------------------------------------------------------------------------------------------------------------------------
//释放之前malloc开辟的空间
void DistroyContact(struct Contact* p)
{
  free(p->data);
  p->data = NULL;
  printf("退出通讯录\n");
}
//保存当前通讯录的成员信息到contactinformation这个文件里
void SaveContact(struct Contact* p)
{
  //以只写的方式打开文件
  FILE* pf = fopen("contactinformation", "wb");
  //判断文件是否打开
  if (pf == NULL)//打开失败的情况
  {
  printf("SaveContact::%s\n", strerror(errno));
  return;
  }
  //开始逐条读取成员信息放到contactinformation文件里
  int i = 0;
  for (i = 0; i < p->size; i++)
  {
  fwrite(p->data + i, sizeof(struct PeoInfo), 1, pf);
  }
  printf("保存成功\n");
  //关闭文件
  fclose(pf);
  pf = NULL;
}


test.c

#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("***** 7.save              0.exit   *****\n");
  printf("****************************************\n");
}
//主函数
int main()
{
  struct Contact con = {0};
  InitContact(&con);
  int input = 0;
  do
  {
  menu();//显示通讯录功能
  printf("请选择:>");
  scanf("%d", &input);
  switch (input)
  {
  case EXIT://0
    SaveContact(&con);
    DistroyContact(&con);
    break;
  case ADD://1
    AddContact(&con);
    break;
  case DEL://2
    DelContact(&con);
    break;
  case SEARCH://3
    SearchContact(&con);
    break;
  case MODIFY://4
    ModifyContact(&con);
    break;
  case SHOW://5
    ShowContact(&con);
    break;
  case SORT://6
    SortContact(&con);
    ShowContact(&con);
    break;
  case SAVE://7
    SaveContact(&con);
    break;
  default:
    printf("选择错误,请重新选择\n");
    break;
  }
  } while (input);
  return 0;
}
相关文章
|
4月前
|
存储 C语言
探索C语言数据结构:利用顺序表完成通讯录的实现
本文介绍了如何使用C语言中的顺序表数据结构实现一个简单的通讯录,包括初始化、添加、删除、查找和保存联系人信息的操作,以及自定义结构体用于存储联系人详细信息。
56 2
|
9月前
|
C语言
C语言——通讯录系统—基于 VS2022
C语言——通讯录系统—基于 VS2022
|
4月前
|
存储 C语言
手把手教你用C语言实现通讯录管理系统
手把手教你用C语言实现通讯录管理系统
|
6月前
|
存储 搜索推荐 算法
【C语言】C语言—通讯录管理系统(源码)【独一无二】
【C语言】C语言—通讯录管理系统(源码)【独一无二】
116 2
|
6月前
|
存储 数据可视化 C语言
【C语言】C语言 手机通讯录系统的设计 (源码+数据+论文)【独一无二】
【C语言】C语言 手机通讯录系统的设计 (源码+数据+论文)【独一无二】
|
8月前
|
机器学习/深度学习 搜索推荐 程序员
C语言实现个人通讯录(功能优化)-2
C语言实现个人通讯录(功能优化)
|
8月前
|
存储 C语言 索引
C语言实现个人通讯录(功能优化)-1
C语言实现个人通讯录(功能优化)
C语言实现个人通讯录(功能优化)-1
|
8月前
|
C语言
C语言学习记录——通讯录(静态内存)
C语言学习记录——通讯录(静态内存)
44 2
|
9月前
|
存储 C语言
C语言实现通讯录
C语言实现通讯录
48 2
|
9月前
|
存储 C语言
C语言实验-动态顺序表实现简易通讯录(二)
在这个C语言实验中,你将实现一个简单的通讯录,它使用动态顺序表来存储联系人信息。
74 2