26.通讯录的实现

简介: 26.通讯录的实现

让我们来看看如何通过顺序表来实现数据结构呢,可以参考下图思路

使用头文件时,容易出现的一个问题,关于解决头文件的嵌套问题,要加 #prama once哦

具体代码如下:

SeqList.h

增强可读性,传入的为结构体指针了

#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include"Contact.h"
 
 
typedef Info SLDataType;
 
typedef struct SeqList
{
  SLDataType* arr; //存储数据的底层结构
  int capacity;    //记录顺序表的空间大小
  int size;        //记录顺序表当前有效的数据个数
}SL;
 
//typedef struct SeqList SL;
 
//初始化和销毁
void SLInit(SL* ps);
void SLDestroy(SL* ps);
void SLPrint(SL* ps); //保持接口一致性
 
//顺序表的头部/尾部插入
void SLPushBack(SL* ps, SLDataType x);
void SLPushFront(SL* ps, SLDataType x);
 
//顺序表的头部/尾部删除
void SLPopBack(SL* ps);
void SLPopFront(SL* ps);
 
//指定位置之前插入数据
//删除指定位置数据
void SLInsert(SL* ps, int pos, SLDataType x);
void SLErase(SL* ps, int pos);
//int SLFind(SL* ps, SLDataType x);

Contact.h

注意:

代码:

#pragma once
 
#define NAME_MAX 100
#define GENDER_MAX 10
#define TEL_MAX 12
#define ADDR_MAX 100
 
//通讯录数据类型
typedef struct PersonInfo
{
  char name[NAME_MAX];
  int age;
  char gender[GENDER_MAX];
  char tel[TEL_MAX];
  char addr[ADDR_MAX];
}Info;
 
//使用顺序表的前置声明
struct SeqList;
 
typedef struct SeqList Contact;
 
//通讯里提供的操作
 
//通讯录的初始化和销毁
void ContactInit(Contact* pcon);//实际初始化的还是顺序表
void ContactDesTroy(Contact* pcon);
 
//增加、删除、修改、查找、查看通讯录
void ContactAdd(Contact* pcon);
void ContactDel(Contact* pcon);
void ContactModify(Contact* pcon);
void ContactFind(Contact* pcon);
void ContactShow(Contact* pcon);

SeqList.c

借用一下上一篇博客顺序表的代码~

#include"SeqList.h"
//初始化和销毁
void SLInit(SL* ps) {
  ps->arr = NULL; //不是int 而是Info类型 
  ps->size = ps->capacity = 0;
}
 
void SLCheckCapacity(SL* ps) {
  if (ps->size == ps->capacity) {
    int newCapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity;
    SLDataType* tmp = (SLDataType*)realloc(ps->arr, newCapacity * sizeof(SLDataType));
    if (tmp == NULL) {
      perror("realloc fail!");
      exit(1);
    }
    //扩容成功
    ps->arr = tmp;
    ps->capacity = newCapacity;
  }
}
 
//顺序表的头部/尾部插入
void SLPushBack(SL* ps, SLDataType x) {
  //断言--粗暴的解决方式
  //assert(ps != NULL);
  assert(ps);
 
  //if判断--温柔的解决方式
  //if (ps == NULL) {
  //  return;
  //}
 
  //空间不够,扩容
  SLCheckCapacity(ps);
 
  //空间足够,直接插入
  ps->arr[ps->size++] = x;
  //ps->size++;
}
void SLPushFront(SL* ps, SLDataType x) {
  assert(ps);
 
  //判断是否扩容
  SLCheckCapacity(ps);
 
  //旧数据往后挪动一位
  for (int i = ps->size; i > 0; i--) //i = 1
  {
    ps->arr[i] = ps->arr[i - 1]; //ps->arr[1] = ps->arr[0]
  }
  ps->arr[0] = x;
  ps->size++;
}
 
//顺序表的头部/尾部删除
void SLPopBack(SL* ps) {
  assert(ps);
  assert(ps->size);
 
  //顺序表不为空
  //ps->arr[ps->size - 1] = -1;
  ps->size--;
}
void SLPopFront(SL* ps) {
  assert(ps);
  assert(ps->size);
 
  //不为空执行挪动操作
  for (int i = 0; i < ps->size-1 ;i++)
  {
    ps->arr[i] = ps->arr[i + 1];
  }
  ps->size--;
}
 
//指定位置之前插入数据
void SLInsert(SL* ps, int pos, SLDataType x) {
  assert(ps);
  assert(pos >= 0 && pos <= ps->size);
  
  SLCheckCapacity(ps);
 
  //pos及之后的数据往后挪动一位,pos空出来
  for (int i = ps->size; i > pos ;i--)
  {
    ps->arr[i] = ps->arr[i - 1]; //ps->arr[pos+1] = ps->arr[pos]
  }
  ps->arr[pos] = x;
  ps->size++;
}
//删除指定位置数据
void SLErase(SL* ps, int pos) {
  assert(ps);
  assert(pos >= 0 && pos < ps->size); 
 
  //pos以后的数据往前挪动一位
  for (int i = pos;i < ps->size-1;i++)
  {
    ps->arr[i] = ps->arr[i + 1];//ps->arr[i-2] = ps->arr[i-1];
  }
  ps->size--;
}
 
 
 
void SLDestroy(SL* ps) {
  assert(ps);
 
  if (ps->arr) {
    free(ps->arr);
  }
  ps->arr = NULL;
  ps->size = ps->capacity = 0;
}
 
void SLPrint(SL* ps) {
  for (int i = 0; i < ps->size; i++)
  {
    printf("%d ", ps->arr[i]);
  }
  printf("\n");
}

Contact.c

有 #prama once 所以不用太担心头文件嵌套啦,放心大胆的都带上

#include"Contact.h"
#include"SeqList.h"
 
//通讯录的初始化和销毁
//SL* ps
void ContactInit(Contact* pcon) {
  SLInit(pcon);
}
 
void ContactDesTroy(Contact* pcon) {
  SLDestroy(pcon);
}
 
//增加、删除、修改、查找、查看通讯录
void ContactAdd(Contact* pcon) {
  //创建联系人结构体变量
  Info info;
 
  printf("请输入联系人姓名:\n");
  scanf("%s", info.name);
  printf("请输入联系人年龄:\n");
  scanf("%d", &info.age);
  printf("请输入联系人性别:\n");
  scanf("%s", info.gender);
  printf("请输入联系人电话:\n");
  scanf("%s", info.tel);
  printf("请输入联系人住址:\n");
  scanf("%s", info.addr);
 
  //保存数据到通讯录(顺序表)
  SLPushBack(pcon, info);
}
 
int FindByName(Contact* pcon, char name[]) {
  for (int i = 0; i < pcon->size; i++)
  {
    if (strcmp(pcon->arr[i].name, name) == 0) {
      //找到了
      return i;
    }
  }
  return -1;
}
 
 
void ContactDel(Contact* pcon) {
  //删除之前一定要先查找
  //找到了,可以删除
  //找不到,不能执行删除
  printf("请输入要删除的联系人姓名:\n");
  char name[NAME_MAX];
  scanf("%s", name);
 
  int findIndex = FindByName(pcon, name);
  if (findIndex < 0) {
    printf("要删除的联系人不存在!\n");
    return;
  }
  //执行删除操作
  SLErase(pcon, findIndex);
  printf("联系人删除成功!\n");
}
 
void ContactModify(Contact* pcon) {
  //修改之前要先查找
  //找到了,执行修改操作
  //没有找到,不能执行修改操作
 
  char name[NAME_MAX];
  printf("请输入要修改的联系人姓名:\n");
  scanf("%s", name);
 
  int findIndex = FindByName(pcon, name);
  if (findIndex < 0) {
    printf("要修改的联系人不存在!\n");
    return;
  }
  //找到了,执行修改操作
  //int arr[10]   int arr[findIndex] = 100
  printf("请输入姓名:\n");
  scanf("%s", pcon->arr[findIndex].name);
  printf("请输入年龄:\n");
  scanf("%d", &pcon->arr[findIndex].age);
  printf("请输入性别:\n");
  scanf("%s", pcon->arr[findIndex].gender);
  printf("请输入电话:\n");
  scanf("%s", pcon->arr[findIndex].tel);
  printf("请输入地址:\n");
  scanf("%s", pcon->arr[findIndex].addr);
 
  printf("联系人修改成功!\n");
}
 
void ContactShow(Contact* pcon) {
  //格式大家下去感兴趣去调整
  printf("%s %s %s %s %s\n", "姓名", "性别", "年龄", "电话", "住址");
 
  for (int i = 0; i < pcon->size; i++)
  {
    printf("%s %s %d %s %s\n",
      pcon->arr[i].name,
      pcon->arr[i].gender,
      pcon->arr[i].age,
      pcon->arr[i].tel,
      pcon->arr[i].addr
      );
  }
}
 
void ContactFind(Contact* pcon) {
  char name[NAME_MAX];
  printf("请输入要查找的用户姓名:\n");
  scanf("%s", name);
 
  int findIndex = FindByName(pcon, name);
  if (findIndex < 0) {
    printf("该联系人不存在!\n");
    return;
  }
  //找到了,打印一下查找的联系人信息
  printf("%s %s %s %s %s\n", "姓名", "性别", "年龄", "电话", "住址");
  printf("%s %s %d %s %s\n",
    pcon->arr[findIndex].name,
    pcon->arr[findIndex].gender,
    pcon->arr[findIndex].age,
    pcon->arr[findIndex].tel,
    pcon->arr[findIndex].addr
  );
}

ConTest.c

//#include"Contact.h"  //在SeqList.h文件中已经包了Contact.h
#include"SeqList.h"
 
//通讯录菜单
void menu() {
  printf("*****************通讯录***************\n");
  printf("*******1.添加联系人  2.删除联系人*****\n");//ctrl+d
  printf("*******3.修改联系人  4.查找联系人*****\n");//ctrl+d
  printf("*******5.查看通讯录  0.  退 出  ******\n");//ctrl+d
  printf("**************************************\n");
}
int main()
{
  int op = -1;
  //创建通讯录结构对象
  Contact con;
  ContactInit(&con);
  do {
    menu();
    printf("请选择您的操作:\n");
    scanf("%d", &op);
 
    switch (op)
    {
    case 1:
      //添加联系人
      ContactAdd(&con);
      break;
    case 2:
      //删除联系人
      ContactDel(&con);
      break;
    case 3:
      //修改联系人
      ContactModify(&con);
      break;
    case 4:
      //查找联系人
      ContactFind(&con);
      break;
    case 5:
      //查看通讯录
      ContactShow(&con);
      break;
    case 0:
      //退出通讯录
      printf("通讯录退出中...\n");
      break;
    default:
      break;
    }
    
  } while (op != 0);
  //销毁通讯录
  ContactDesTroy(&con);
  return 0;
}


相关文章
简易实现通讯录(1.0)
简易实现通讯录(1.0)
50 0
简易实现通讯录(2.0)
简易实现通讯录(2.0)
53 0
|
7月前
|
数据管理
【通讯录项目 (3 / 3)】基于顺序表的通讯录实现——通讯录项目实现
通讯录项目我们实现了大部分内容,接下来你可以自行探索,丰富功能。
56 0
|
存储
通讯录实现上
通讯录实现上
通讯录实现上
|
C语言
【纯C实现简易通讯录】
【纯C实现简易通讯录】
63 0
学C的第三十一天【通讯录的实现】-2
9 . 函数 SearchContact() 的声明 -- 查找指定通讯录成员信息: 参数接收 -- const Contact* pc (只是打印,不用修改所以设置为常量) 返回类型 -- void
121 0
|
7月前
通讯录实现
通讯录实现
48 0
|
7月前
|
存储 Java C++
通讯录的实现
通讯录的实现
69 0
|
C++
简易通讯录的实现(上)
简易通讯录的实现(上)
51 0
简易通讯录的实现(下)
简易通讯录的实现(下)
54 0