【C版本】静态通讯录与动态通讯录的实现,以及各自所存在的缺陷对比。(含所有原码)

简介: 【C版本】静态通讯录与动态通讯录的实现,以及各自所存在的缺陷对比。(含所有原码)

https://ucc.alicdn.com/images/user-upload-01/84df24f1adf5441fb039518d9ecd5876.gif#pic_center

目录


静态版本通讯录

前期思路

具体实现

1、框架

2、初始化通讯录

3、增加联系人

4、显示已有联系人

5、查找联系人

6、删除指定联系人

7、排序联系人

8、修改联系人信息

9、清空联系人

静态版本通讯录存在的缺陷

动态版本通讯录(静态版本的部分功能发生改动)

初始化

增加联系人

退出程序(释放空间)

动态版本通讯录存在的缺陷

动态通讯录原码

静态版本通讯录


前期思路

与之前的扫雷以及三子棋的实现方式是一样的,创建两个源文件,一个用来测试,一个用来存放函数定义,再创建一个头文件,用来存放函数声明。接下来是着手实现通讯录。

首先要有一个大概的框架,并且要明确即将实现的通讯录的功能,最基本的即增删查改,然后对这些功能进行进一步的细化实现,并且我们知道,人的信息是一个比较复杂的对象,不可能用一句话就概括,所以就用到了之前学过的结构体,一个结构体用来存放联系人,另一个用来存放联系人对应的的基本信息。

最后我们要知道一点,就是一口吃不成一个大胖子,功能实现的过程是一步步来的。


1.png


具体实现

1、框架

首先创建一个用来存放联系人以及记录联系人个数的结构体,然后把联系人的信息也存放在这个结构体中,如下:


//联系人信息
struct message
{
  //姓名
  char name[NAME];
  //性别
  char sex[SEX];
  //电话
  char tele[TELE];
  //住址
  char addr[ADDR];
  //年龄
  int age;
};
//通讯录
struct contact
{
  //存放联系人的数组,数组里存放的元素类型是结构体类型,即存放联系人的信息。
  struct message data[MAX];
  //记录联系人的个数
  int sz;
};


这里我们可以用define来定义一个常量,这样后续修改起来也比较容易,不要把程序写死。


#define NAME 20//姓名
#define SEX 5//性别
#define TELE 12//电话
#define ADDR 30//住址
#define MAX 100//联系人最大个数


然后在.c的测试文件里书写菜单栏。


(个人建议:像这种,不算是特别复杂的对象,就可以提前制定一个框架,也就是菜单栏,但是后面学到数据结构的时候,建议最后再添加,因为菜单栏的存在,会使一些调试比较麻烦。)


一个基本的框架,满足上面提到的一些功能,实现起来也比较容易,用一个简单的do while即可,如下:


#include"contact_.h"
//菜单栏
void menu()
{
  printf("--------------------------------------------------------------\n");
  printf("---------   1、增加联系人       2、删除指定联系人    ---------\n");
  printf("---------   3、修改联系人信息   4、查找联系人        ---------\n");
  printf("---------   5、排序联系人       6、显示已有联系人    ---------\n");
  printf("---------   0、退出             7、清空联系人        ---------\n");
  printf("--------------------------------------------------------------\n");
}
int main()
{
  int input = 0;
  //创建通讯录
  struct contact con;
  //初始化通讯录
  Init_contact(&con);
  do
  {
  menu();
  printf("请选择:->");
  scanf("%d", &input);
  system("cls");
  switch (input)
  {
  case 1:
    //增加联系人
    Add_contact(&con);
    break;
  case 2:
    //删除联系人
    Show_contact(&con);
    Dele_contact(&con);
    break;
  case 3:
    //修改联系人信息
    Show_contact(&con);
    revise_contact(&con);
    break;
  case 4:
    //查找联系人信息
    Find_contact(&con);
    break;
  case 5:
    //排序联系人信息
    Show_contact(&con);
    Sort_contact(&con);
    break;
  case 6:
    //显示联系人
    system("cls");
    Show_contact(&con);
    printf("\n");
    break;
  case 7:
    //清空联系人
    Clear(&con);
    break;
  case 0:
    break;
  default:
    printf("输入错误!\n");
    break;
  }
  } while (input);
  return 0;
}


2、初始化通讯录

与扫雷游戏一样,首先先把通讯录初始化,然后再存放信息。实现起来也很简单。


//初始化通讯录
void Init_contact(struct contact* p)
{
  assert(p);
  p->sz = 0;//sz是记录联系人个数的变量
  memset(p->data, 0, MAX * sizeof(struct message));
  //p指向的data数组(存放联系人的)里,把MAX个联系人信息都置为0
  //memset是一个内存函数,修改存储在内存的数据
}


3、增加联系人

增加一个联系人,即增加姓名、性别、电话等信息,把这些信息输入在结构体数组中对应得结构体成员即可,如下:


//增加联系人
void Add_contact(struct contact* p)
{
  assert(p);
  if (p->sz == MAX)
  printf("联系人已满!!!\n");
  printf("请输入姓名:->");
  scanf("%s", p->data[p->sz].name);
  printf("请输入性别:->");
  scanf("%s", p->data[p->sz].sex);
  printf("请输入电话:->");
  scanf("%s", p->data[p->sz].tele);
  printf("请输入住址:->");
  scanf("%s", p->data[p->sz].addr);
  printf("请输入年龄:->");
  scanf("%d", &(p->data[p->sz].age));
  system("cls");
  //清屏
  printf("增加成功!\n");
  printf("\n");
  p->sz++;//每增加一个,sz也跟着++
}


2.png


说白了这块知识点就是结构体成员的访问,只不过这里访问了两次


4、显示已有联系人

//显示联系人
void Show_contact(const struct contact* p)
{
  assert(p);
  int i = 0;
  //为了显示出来更加有美感,先打印一行基本信息
  printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5s\n", "姓名", "性别", "电话", "住址", "年龄");
  //循环打印即可,sz记录目前联系人个数
  for (i = 0; i < p->sz; i++)
  {
      // 一一对应即可
  printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5d\n", p->data[i].name,
    p->data[i].sex,
    p->data[i].tele,
    p->data[i].addr,
    p->data[i].age);
  }
}


实现完成如下;


3.png


5、查找联系人

查找联系人的实现也很简单,定义一个函数,遍历整个data数组,如果不存在返回-1,打印不存在,存在就返回1,打印该联系人信息。如下:


//这里遍历整个数组,如果不存在,则返回-1,存在返回1
int find_name(const struct contact* p, char arr[])
{
  assert(p);
  int i = 0;
  for (i = 0; i < p->sz; i++)
  {
  if (0 == strcmp(p->data[i].name, arr))
    return i;
  }
  return -1;
}
//查找联系人信息
void Find_contact(const struct contact* p)
{
  assert(p);
  char del_name[NAME];
  printf("请输入要查找联系人的姓名:->");
  scanf("%s", del_name);
  system("cls");
  //查找该联系人
  int ret = find_name(p, del_name);
  if (ret == -1)
  printf("查无此人!\n");
  //返回值为1,打印出该联系人信息即可
  else
  {
  printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5s\n", "姓名", "性别", "电话", "住址", "年龄");
  printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5d\n", p->data[ret].name,
    p->data[ret].sex,
    p->data[ret].tele,
    p->data[ret].addr,
    p->data[ret].age);
  }
}


6、删除指定联系人

实现删除的功能也很简单,由于数组在内存中是连续存放的,只需要后面的覆盖即可,原理如下:


4.png


具体用代码实现如下:


//删除联系人
void Dele_contact(struct contact* p)
{
  assert(p);
  char del_name[NAME];
  printf("请输入要删除联系人的姓名:->");
  scanf("%s", del_name);
  //查找该联系人
  int ret = find_name(p, del_name);
  if (ret == -1)
  printf("查无此人!\n");
  else
  {
  int j = 0;
  for (j = ret; j < p->sz - 1; j++)
  {
    p->data[j] = p->data[j + 1];//后面的覆盖前面的
  }
  p->sz--;
  system("cls");
  printf("删除成功!\n");
  printf("\n");
  }
}


7、排序联系人

这里我又新增了一步,就是可以实现按姓名、住址、年龄、性别排序,用qsort即可实现,不懂的可以去翻看我前面的指针进阶文章,里面有介绍。具体实现如下:


//排序菜单
void menu_sort()
{
  printf("******   1、姓名   ******\n");
  printf("******   2、住址   ******\n");
  printf("******   3、年龄   ******\n");
  printf("******   4、性别   ******\n");
  printf("******   0、退出   ******\n");
}
//姓名排序
int cmp_name(const void* e1, const void* e2)
{
  return strcmp(((struct message*)e1)->name, ((struct message*)e2)->name);
}
//住址排序
int cmp_addr(const void* e1, const void* e2)
{
  return strcmp(((struct message*)e1)->addr, ((struct message*)e2)->addr);
}
//年龄排序
int cmp_age(const void* e1, const void* e2)
{
  return ((struct message*)e1)->age - ((struct message*)e2)->age;
}
//性别排序
int cmp_sex(const void* e1, const void* e2)
{
  return strcmp(((struct message*)e1)->sex, ((struct message*)e2)->sex);
}
//排序联系人
void Sort_contact(struct contact* p)
{
  int s = 0;
  do
  {
  //排序菜单
  menu_sort();
  printf("请选择排序类型:->");
  scanf("%d", &s);
  system("cls");
  switch (s)
  {
  case 1:
    qsort(p->data, p->sz, sizeof(struct message), cmp_name);
    printf("排序成功!\n");
    break;
  case 2:
    qsort(p->data, p->sz, sizeof(struct message), cmp_addr);
    printf("排序成功!\n");
    break;
  case 3:
    qsort(p->data, p->sz, sizeof(struct message), cmp_age);
    printf("排序成功!\n");
    break;
  case 4:
    qsort(p->data, p->sz, sizeof(struct message), cmp_sex);
    printf("排序成功!\n");
    break;
  case 0:
    printf("退出排序\n");
    break;
  default:
    printf("输入有误!\n");
    break;
  }
  } while (s);
}


8、修改联系人信息

修改联系人信息,首先找到这个联系人,然后再进行修改,这里我是可以把信息有选择性的修改,实现起来也很简单


//修改菜单栏
void menu_()
{
  printf("***********************************\n");
  printf("******   1、修改联系人姓名   ******\n");
  printf("******   2、修改联系人电话   ******\n");
  printf("******   3、修改联系人年龄   ******\n");
  printf("******   4、修改联系人住址   ******\n");
  printf("******   5、修改联系人性别   ******\n");
  printf("******   0、返回主菜单       ******\n");
}
//修改联系人信息
void revise_contact(struct contact* p)
{
  assert(p);
  char del_name[NAME];
  printf("请输入要修改信息的联系人的姓名:->");
  scanf("%s", del_name);
  int ret = find_name(p, del_name);
  if (ret == -1)
  {
  printf("查无此人!\n");
  printf("\n");
  }
  else
  {
  int in_put = 0;
  do
  {
    menu_();
    scanf("%d", &in_put);
    switch (in_put)
    {
    case 1:
    printf("请输入修改后的姓名:->");
    scanf("%s", p->data[ret].name);
    system("cls");
    printf("姓名修改成功!\n");
    break;
    case 2:
    printf("请输入修改后的电话:->");
    scanf("%s", p->data[ret].tele);
    system("cls");
    printf("电话修改成功!\n");
    break;
    case 3:
    printf("请输入修改后的年龄:->");
    scanf("%d", &(p->data[ret].age));
    system("cls");
    printf("年龄修改成功!\n");
    break;
    case 4:
    printf("请输入修改后的住址:->");
    scanf("%s", p->data[ret].addr);
    system("cls");
    printf("住址修改成功!\n");
    break;
    case 5:
    printf("请输入修改后的性别:->");
    scanf("%s", p->data[ret].sex);
    system("cls");
    printf("性别修改成功!\n");
    break;
    case 0:
    printf("取消修改!\n");
    break;
    default:
    printf("输入错误!\n");
    break;
    }
  } while (in_put);
  }
}


9、清空联系人

这一步是最容易实现的一步,只需要置空sz即可,到这里也算是苦尽甘来。


//清空联系人
void Clear(struct contact* p)
{
  p->sz = 0;
  printf("清空成功!\n");
}


至此。一个功能齐全的通讯录实现完毕。


静态版本通讯录存在的缺陷

我们看到,上面的版本我们定义了数组大小为100,也就是能存放100人的信息,data空间是已经开辟了的,假如我们要存放第101个人的信息呢?这不就存不下了,有的铁子可能说,那我们可以定义为1000呀,你总不能有这么多联系人用来存放吧。

确实如此,但是后面的空间不就浪费了。那有什么办法可以实现按照我们的需求来开辟合适的空间呢?答案是有的,就是后面的动态内存版本。


动态版本通讯录(静态版本的部分功能发生改动)


动态版本的通讯录是在静态版本上进行的一次优化,即实现按照需求开辟空间。和之前版本有所不同,不用一个结构体数组来存放联系人,而是用了一个结构体指针。如下:


#define NAME 20
#define SEX 5
#define TELE 12
#define ADDR 30
#define DEFAULT_SZ 3//初始容量
#define INC_SZ 2//扩容
//联系人信息
struct message
{
  //姓名
  char name[NAME];
  //性别
  char sex[SEX];
  //电话
  char tele[TELE];
  //住址
  char addr[ADDR];
  //年龄
  int age;
};
//通讯录
struct contact
{
  struct message* data;//结构体指针
  int sz;//个数
  int capacity;//通讯录容量
};



由于是动态版本的,所以在初始化、增加联系人、以及最后的退出里,有了一定的修改,别的只要不涉及增加空间的操作,都与静态版本相同。这里就不一一再写了。


初始化

//初始化通讯录
void Init_contact(struct contact* p)
{
  assert(p);
  //开辟空间
  p->data =(struct message*) malloc(DEFAULT_SZ * sizeof(struct message));
  //假如开辟失败,报错
  if (p->data == NULL)
  {
  printf("%s\n", strerror(errno));
  return;
  }
  p->sz = 0;
  p->capacity = DEFAULT_SZ;
}


这里的capacity是当前通讯录的容量,sz是记录当前联系人的数量,当数量==容量时,就要进行扩容。

所以这里又增加了一个用来判断是否扩容的函数,这个函数算是动态版本通讯录的核心函数了


//是否判断增容
int check_capacity(struct contact*p)
{
  //当联系人个数 == 通讯录容量时,增容INC_SZ个内存空间
  if (p->sz == p->capacity)
  {
  struct message* ptr = (struct message*)realloc(p->data, (p->capacity + INC_SZ) * sizeof(struct message));
  if (ptr == NULL)//判断是否增容失败
  {
    printf("%s\n", strerror(errno));
    return 0;
  }
  else
  {
    p->data = ptr;//增容成功,data就指向这块新开辟的空间
    p->capacity += INC_SZ;//容量+=INC_SZ
    //printf("增容成功!\n");
    return 1;
  }
  }
  //不需要增容
  else
  return 1;
}


增加联系人

这里的增加联系人,就要在增加之前进行判断,空间是否需要扩容,如下:

check_capacity时上面写的用来判断知否增容的函数。


//增加联系人
void Add_contact(struct contact* p)
{
  assert(p);
  if (0 == check_capacity(p))
  {
  printf("%s\n", strerror(errno));
  return;
  }
  printf("请输入姓名:->");
  scanf("%s", p->data[p->sz].name);
  printf("请输入性别:->");
  scanf("%s", p->data[p->sz].sex);
  printf("请输入电话:->");
  scanf("%s", p->data[p->sz].tele);
  printf("请输入住址:->");
  scanf("%s", p->data[p->sz].addr);
  printf("请输入年龄:->");
  scanf("%d", &(p->data[p->sz].age));
  system("cls");
  printf("增加成功!\n");
  printf("\n");
  p->sz++;
}


退出程序(释放空间)

malloc这些动态内存管理函数,都是与free成对出现的,这里的释放空间就是需要在程序退出的时候,进行释放,否则造成内存泄漏问题。这里free后置空即可。


//释放空间
void Destory_contact(struct contact* p)
{
  free(p->data);
  p->data=NULL;
  p->sz = 0;
  p->capacity = 0;
}



至此,动态版本通讯录实现完毕,就是在静态版本的基础上进行了修改,主要涉及到了增容问题,处理好即可。


动态版本通讯录存在的缺陷


唯一的缺陷就在于不能把信息保存下来,也就是说,当下次打开程序的时候,上一次写的都没了,也就是说,这是一个“一次性”的通讯录。当然,后面还会有进一步的改动,实现真正意义上的通讯录,即可以把每次的信息保存下来,后续会书写。


动态通讯录原码


头文件


#pragma once
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
#include<Windows.h>
#include<errno.h>
#define NAME 20
#define SEX 5
#define TELE 12
#define ADDR 30
#define DEFAULT_SZ 3//初始容量
#define INC_SZ 2//扩容
//联系人信息
struct message
{
  //姓名
  char name[NAME];
  //性别
  char sex[SEX];
  //电话
  char tele[TELE];
  //住址
  char addr[ADDR];
  //年龄
  int age;
};
//通讯录
struct contact
{
  struct message* data;
  int sz;//个数
  int capacity;//通讯录容量
};
//初始化通讯录
void Init_contact(struct contact* p);
//动态增加联系人
void Add_contact(struct contact* p);
//显示联系人
void Show_contact(const struct contact* p);
//删除联系人
void Dele_contact(struct contact* p);
//修改联系人信息
void revise_contact(struct contact* p);
//查找联系人信息
void Find_contact(const struct contact* p);
//排序联系人
void Sort_contact(struct contact* p);
//清空联系人
void Clean(struct contact* p);
//释放空间
void Destory_contact(struct contact* p);


.c测试源文件


#define _CRT_SECURE_NO_WARNINGS 1
#include"contact_dynamic.h"
void menu()
{
  printf("--------------------------------------------------------------\n");
  printf("---------   1、增加联系人       2、删除指定联系人    ---------\n");
  printf("---------   3、修改联系人信息   4、查找联系人        ---------\n");
  printf("---------   5、排序联系人       6、显示已有联系人    ---------\n");
  printf("---------   0、退出             7、清空联系人        ---------\n");
  printf("--------------------------------------------------------------\n");
}
int main()
{
  int input = 0;
  //创建通讯录
  struct contact con;
  //初始化通讯录
  Init_contact(&con);
  do
  {
  menu();
  printf("请选择:->");
  scanf("%d", &input);
  system("cls");
  switch (input)
  {
  case 1:
    //增加联系人
    Add_contact(&con);
    break;
  case 2:
    //删除联系人
    Show_contact(&con);
    Dele_contact(&con);
    break;
  case 3:
    //修改联系人信息
    Show_contact(&con);
    revise_contact(&con);
    break;
  case 4:
    //查找联系人信息
    Find_contact(&con);
    break;
  case 5:
    //排序联系人信息
    Show_contact(&con);
    Sort_contact(&con);
    break;
  case 6:
    //显示联系人
    system("cls");
    Show_contact(&con);
    printf("\n");
    break;
  case 7:
    //清空联系人
    Clean(&con);
    break;
  case 0:
    //释放空间
    Destory_contact(&con);
    break;
  default:
    printf("输入错误!\n");
    break;
  }
  } while (input);
  return 0;
}


.c存放函数定义的源文件


#include"contact_dynamic.h"
//初始化通讯录
void Init_contact(struct contact* p)
{
  assert(p);
  //开辟空间
  p->data =(struct message*) malloc(DEFAULT_SZ * sizeof(struct message));
  //假如开辟失败,报错
  if (p->data == NULL)
  {
  printf("%s\n", strerror(errno));
  return;
  }
  p->sz = 0;
  p->capacity = DEFAULT_SZ;
}
//是否判断增容
int check_capacity(struct contact*p)
{
  //当联系人个数 == 通讯录容量时,增容INC_SZ个内存空间
  if (p->sz == p->capacity)
  {
  struct message* ptr = (struct message*)realloc(p->data, (p->capacity + INC_SZ) * sizeof(struct message));
  if (ptr == NULL)//判断是否增容失败
  {
    printf("%s\n", strerror(errno));
    return 0;
  }
  else
  {
    p->data = ptr;//增容成功,data就指向这块新开辟的空间
    p->capacity += INC_SZ;//容量+=INC_SZ
    //printf("增容成功!\n");
    return 1;
  }
  }
  //不需要增容
  else
  return 1;
}
//增加联系人
void Add_contact(struct contact* p)
{
  assert(p);
  if (0 == check_capacity(p))
  {
  printf("%s\n", strerror(errno));
  return;
  }
  printf("请输入姓名:->");
  scanf("%s", p->data[p->sz].name);
  printf("请输入性别:->");
  scanf("%s", p->data[p->sz].sex);
  printf("请输入电话:->");
  scanf("%s", p->data[p->sz].tele);
  printf("请输入住址:->");
  scanf("%s", p->data[p->sz].addr);
  printf("请输入年龄:->");
  scanf("%d", &(p->data[p->sz].age));
  system("cls");
  printf("增加成功!\n");
  printf("\n");
  p->sz++;
}
//显示联系人
void Show_contact(const struct contact* p)
{
  assert(p);
  int i = 0;
  printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5s\n", "姓名", "性别", "电话", "住址", "年龄");
  for (i = 0; i < p->sz; i++)
  {
  printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5d\n", p->data[i].name,
    p->data[i].sex,
    p->data[i].tele,
    p->data[i].addr,
    p->data[i].age);
  }
}
int find_name(const struct contact* p, char arr[])
{
  assert(p);
  int i = 0;
  for (i = 0; i < p->sz; i++)
  {
  if (0 == strcmp(p->data[i].name, arr))
    return i;
  }
  return -1;
}
//删除联系人
void Dele_contact(struct contact* p)
{
  assert(p);
  char del_name[NAME];
  printf("请输入要删除联系人的姓名:->");
  scanf("%s", del_name);
  //查找该联系人
  int ret = find_name(p, del_name);
  if (ret == -1)
  printf("查无此人!\n");
  else
  {
  int j = 0;
  for (j = ret; j < p->sz - 1; j++)
  {
    p->data[j] = p->data[j + 1];
  }
  p->sz--;
  system("cls");
  printf("删除成功!\n");
  printf("\n");
  }
}
//修改菜单栏
void menu_()
{
  printf("***********************************\n");
  printf("******   1、修改联系人姓名   ******\n");
  printf("******   2、修改联系人电话   ******\n");
  printf("******   3、修改联系人年龄   ******\n");
  printf("******   4、修改联系人住址   ******\n");
  printf("******   5、修改联系人性别   ******\n");
  printf("******   0、返回主菜单       ******\n");
}
//修改联系人信息
void revise_contact(struct contact* p)
{
  assert(p);
  char del_name[NAME];
  printf("请输入要修改信息的联系人的姓名:->");
  scanf("%s", del_name);
  int ret = find_name(p, del_name);
  if (ret == -1)
  {
  printf("查无此人!\n");
  printf("\n");
  }
  else
  {
  int in_put = 0;
  do
  {
    menu_();
    scanf("%d", &in_put);
    switch (in_put)
    {
    case 1:
    printf("请输入修改后的姓名:->");
    scanf("%s", p->data[ret].name);
    system("cls");
    printf("姓名修改成功!\n");
    break;
    case 2:
    printf("请输入修改后的电话:->");
    scanf("%s", p->data[ret].tele);
    system("cls");
    printf("电话修改成功!\n");
    break;
    case 3:
    printf("请输入修改后的年龄:->");
    scanf("%d", &(p->data[ret].age));
    system("cls");
    printf("年龄修改成功!\n");
    break;
    case 4:
    printf("请输入修改后的住址:->");
    scanf("%s", p->data[ret].addr);
    system("cls");
    printf("住址修改成功!\n");
    break;
    case 5:
    printf("请输入修改后的性别:->");
    scanf("%s", p->data[ret].sex);
    system("cls");
    printf("性别修改成功!\n");
    break;
    case 0:
    printf("取消修改!\n");
    break;
    default:
    printf("输入错误!\n");
    break;
    }
  } while (in_put);
  }
}
//查找联系人信息
void Find_contact(const struct contact* p)
{
  assert(p);
  char del_name[NAME];
  printf("请输入要查找联系人的姓名:->");
  scanf("%s", del_name);
  system("cls");
  //查找该联系人
  int ret = find_name(p, del_name);
  if (ret == -1)
  printf("查无此人!\n");
  else
  {
  printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5s\n", "姓名", "性别", "电话", "住址", "年龄");
  printf("%-20s\t%-5s\t%-12s\t%-20s\t%-5d\n", p->data[ret].name,
    p->data[ret].sex,
    p->data[ret].tele,
    p->data[ret].addr,
    p->data[ret].age);
  }
}
//排序菜单
void menu_sort()
{
  printf("******   1、姓名   ******\n");
  printf("******   2、住址   ******\n");
  printf("******   3、年龄   ******\n");
  printf("******   4、性别   ******\n");
  printf("******   0、退出   ******\n");
}
//姓名排序
int cmp_name(const void* e1, const void* e2)
{
  return strcmp(((struct message*)e1)->name, ((struct message*)e2)->name);
}
//住址排序
int cmp_addr(const void* e1, const void* e2)
{
  return strcmp(((struct message*)e1)->addr, ((struct message*)e2)->addr);
}
//年龄排序
int cmp_age(const void* e1, const void* e2)
{
  return ((struct message*)e1)->age - ((struct message*)e2)->age;
}
//性别排序
int cmp_sex(const void* e1, const void* e2)
{
  return strcmp(((struct message*)e1)->sex, ((struct message*)e2)->sex);
}
//排序联系人
void Sort_contact(struct contact* p)
{
  int s = 0;
  do
  {
  //排序菜单
  menu_sort();
  printf("请选择排序类型:->");
  scanf("%d", &s);
  system("cls");
  switch (s)
  {
  case 1:
    qsort(p->data, p->sz, sizeof(struct message), cmp_name);
    printf("排序成功!\n");
    break;
  case 2:
    qsort(p->data, p->sz, sizeof(struct message), cmp_addr);
    printf("排序成功!\n");
    break;
  case 3:
    qsort(p->data, p->sz, sizeof(struct message), cmp_age);
    printf("排序成功!\n");
    break;
  case 4:
    qsort(p->data, p->sz, sizeof(struct message), cmp_sex);
    printf("排序成功!\n");
    break;
  case 0:
    printf("退出排序\n");
    break;
  default:
    printf("输入有误!\n");
    break;
  }
  } while (s);
}
//清空联系人
void Clean(struct contact* p)
{
  p->sz = 0;
  printf("清空成功!\n");
}
//释放空间
void Destory_contact(struct contact* p)
{
  free(p->data);
  p->data=NULL;
  p->sz = 0;
  p->capacity = 0;
}


end

生活原本沉闷,但跑起来就会有风!


相关文章
|
6月前
|
存储
通讯录(动态实现与文件优化版)
通讯录(动态实现与文件优化版)
51 1
【文件版&动态版通讯录】
【文件版&动态版通讯录】
37 0
|
6月前
|
C语言
【C语言】动态内存管理基础知识——动态通讯录,如何实现通讯录容量的动态化
动态内存管理的函数有:malloc,calloc,ralloc,free,本文讲解动态内存函数和使用,如何进行动态内存管理,实现通讯录联系人容量的动态化,对常见动态内存错误进行总结。
72 0
|
存储 C语言
【C语言】通讯录的实现(静态, 动态, 文件)
【C语言】通讯录的实现(静态, 动态, 文件)
|
6月前
|
存储 编译器 C语言
通讯录详解(静态版,动态版,文件版)
通讯录详解(静态版,动态版,文件版)
91 0
|
存储
C实现通讯录(静态版+动态版)(一)
C实现通讯录(静态版+动态版)
41 0
C实现通讯录(静态版+动态版)(一)
C实现通讯录(静态版+动态版)(二)
C实现通讯录(静态版+动态版)
44 0
|
程序员 编译器 C语言
C语言---认识动态内存管理并实现一个动态通讯录:静态通讯录别来沾边
C语言学习——动态内存管理(上)+优化版通讯录+笔试题
通讯录的8种功能的具体实现和整个程序的代码
具体详细讲解看上一个博客(贼细) 1.头文件(声明各种函数和定义各种类型的地方) 2.测试文件(main函数所在,代码开始的地方) 3.函数实现文件(8种功能的具体实现,每一个函数都是独立实现,无嵌套使用) 4.以上你可以写在一个文件中也可以写在不同文件中
|
C语言
通讯录【一】静态版本
通讯录【一】静态版本