【数据结构】----顺序表项目-通讯录

简介: 【数据结构】----顺序表项目-通讯录

顺序表项目-通讯录

顺序表的底层是数组—决定了其作为可以存储不单一数据的结构体的特性—从简单的顺序表变成通讯录

通讯录中不同的数据构成了一个结构体,例如包含联系人的数据:姓名,性别,电话,家庭住址,年龄…

诸如此类。

而我们针对这些通讯录中的数据进行一系列的操作,例如增删查改

然而这些操作实际上已经在顺序表中实现好,我们只需要直接更改函数的名称并且直接调用即可,或者是直接调用已有的函数放在通讯录的操作函数中直接实现。

顺序表和通讯录的相似之处

顺序表的操作

增加

删除

查找

修改

通讯录的操作

增加联系人数据

删除联系人数据

查找联系人数据

修改联系人数据

接下来我们开始进行通讯录项目的理论解析以及代码编写。

项目大纲

每个项目都需要一个框架来支撑它的逻辑结构,我们可以使用来大致阐述项目所需要实现的功能,各个部分的分工等等。如下:

项目的编程实现

下面针对导图的思维来进行具体实现。


具体实现

执行界面 Contest.c

菜单的设计

菜单设计使用一个函数来实现即可。

//通讯录菜单
void menu() {
  printf("*****************通讯录***************\n");
  printf("*******1.添加联系人  2.删除联系人*****\n");
  printf("*******3.修改联系人  4.查找联系人*****\n");
  printf("*******5.查看通讯录  0.  退 出  ******\n");
  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;
}

其中所涉及的函数在后续会进行编写。

通讯录声明 Contest.h

首先我们对所需的数据进行一个长度限制,保证数据的合理性

#define NAME_MAX 100
//#define AGE_MAX
#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;

最后就是通讯里的操作声明

//通讯里提供的操作
//通讯录的初始化和销毁
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);

通讯录操作 Contact.c

初始化
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);
}
查找联系人数据

1.基于其他函数的查找

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;
}

2.独立的查找

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
  );
}
//注:此处的查找使用最简单的查找方式,而在实际过程中我们完全可以使用例如二分法来降低时间复杂度从而提高效率。
//本文章仅介绍关于通讯录的基本操作及代码,可以自行完善。
删除联系人数据
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
      );
  }
}

以上的操作都基于顺序表的对应实现。

当然你也可以基于此发散更多有关的操作,例如对联系人重命名,拉入黑名单(不同于删除)等等,在此只进行基本操作的介绍。

顺序表操作 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--;
}
//在顺序表中查找X
//int SLFind(SL* ps, SLDataType x) {
//  //加上断言对代码的健壮性更好
//  assert(ps);
//  for (int i = 0; i < ps->size; i++)
//  {
//    if (ps->arr[i] == x) {
//      return i;
//    }
//  }
//  return -1;
//}
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");
}

顺序表声明 SeqList.h

#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include"Contact.h"//包含了通讯录的头文件
//静态顺序表
//#define N 100
//struct SeqList
//{
//  SLDataType a[N];
//  int size;
//};
//动态顺序表
//typedef int SLDataType;
typedef Info SLDataType;//注:此处存储的不再是顺序表有关的例如int的基本类型,而是基于通讯录的复合类型
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"的包含,而"Contact.c"中同时包含了"**SeqList.h****以及"Contact.h",这就出现了头文件互相包含的嵌套问题。

解决办法:前置声明:在"Contact.h"中进行前置声明,以保证不会出现反复包含的错误。

//Contact.h
//使用顺序表的前置声明
struct SeqList;
typedef struct SeqList Contact;

所有代码

ConTest.

/#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;
}

Contact.c

#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
  );
}

Contact.h

#pragma once
//#include<stdio.h> //暂时加上
#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--;
}
//在顺序表中查找X
//int SLFind(SL* ps, SLDataType x) {
//  //加上断言对代码的健壮性更好
//  assert(ps);
//  for (int i = 0; i < ps->size; i++)
//  {
//    if (ps->arr[i] == x) {
//      return i;
//    }
//  }
//  return -1;
//}
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");
}

SeqList.h

#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include"Contact.h"
//静态顺序表
//#define N 100
//struct SeqList
//{
//  SLDataType a[N];
//  int size;
//};
//动态顺序表
//typedef int SLDataType;
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);

i < ps->size; i++)

{

printf(“%d “, ps->arr[i]);

}

printf(”\n”);

}

### SeqList.h
```c
#pragma once
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include"Contact.h"
//静态顺序表
//#define N 100
//struct SeqList
//{
//  SLDataType a[N];
//  int size;
//};
//动态顺序表
//typedef int SLDataType;
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);
目录
相关文章
|
12天前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】顺序表的基本运算(头歌实践教学平台习题)【合集】
本文档介绍了线性表的基本运算任务,涵盖顺序表和链表的初始化、销毁、判定是否为空、求长度、输出、查找元素、插入和删除元素等内容。通过C++代码示例详细展示了每一步骤的具体实现方法,并提供了测试说明和通关代码。 主要内容包括: - **任务描述**:实现顺序表的基本运算。 - **相关知识**:介绍线性表的基本概念及操作,如初始化、销毁、判定是否为空表等。 - **具体操作**:详述顺序表和链表的初始化、求长度、输出、查找、插入和删除元素的方法,并附有代码示例。 - **测试说明**:提供测试输入和预期输出,确保代码正确性。 - **通关代码**:给出完整的C++代码实现,帮助完成任务。 文档
27 5
|
26天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 算法 安全
2024重生之回溯数据结构与算法系列学习之顺序表【无论是王道考研人还真爱粉都能包会的;不然别给我家鸽鸽丢脸好嘛?】
顺序表的定义和基本操作之插入;删除;按值查找;按位查找等具体详解步骤以及举例说明
|
2月前
|
存储 C语言
【数据结构】顺序表(c语言实现)(附源码)
本文介绍了线性表和顺序表的基本概念及其实现。线性表是一种有限序列,常见的线性表有顺序表、链表、栈、队列等。顺序表是一种基于连续内存地址存储数据的数据结构,其底层逻辑是数组。文章详细讲解了静态顺序表和动态顺序表的区别,并重点介绍了动态顺序表的实现,包括初始化、销毁、打印、增删查改等操作。最后,文章总结了顺序表的时间复杂度和局限性,并预告了后续关于链表的内容。
98 3
|
2月前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之顺序表习题精讲【无论是王道考研人还真爱粉都能包会的;不然别给我家鸽鸽丢脸好嘛?】
顺序表的定义和基本操作之插入;删除;按值查找;按位查找习题精讲等具体详解步骤以及举例说明
|
3月前
|
存储
数据结构(顺序表)
数据结构(顺序表)
34 0
|
3月前
|
存储 算法
【数据结构】新篇章 -- 顺序表
【数据结构】新篇章 -- 顺序表
26 0
|
2月前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
284 9
|
2月前
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
44 1
|
12天前
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
128 75

热门文章

最新文章