数据结构与算法解析(C语言版)--线性表

news/2024/10/20 13:30:42/

本栏目致力于从0开始使用纯C语言将经典算法转换成能够直接上机运行的程序,以项目的形式详细描述数据存储结构、算法实现和程序运行过程。

参考书目如下:

        《数据结构C语言版-严蔚敏》

        《数据结构算法解析第2版-高一凡》

软件工具:

        dev-cpp


0、准备工作

在项目下创建line.c和line.h文件。

1、线性表操作

1.1 准备工作line.h

定义线性表数组的长度和扩容量

// 线性表长度
#define LIST_INIT_SIZE 100
// 线性表扩容量
#define LISTINCREMENT 10

定义线性表结构体,存储线性表其实地址和线性表元素个数和线性表总长度。

typedef struct
{ElemType *elem;    // 线性表的起始地址 int length;       // 表中元素个数 int listsize;     // 表的总长度 
}SqList;

1.2 初始化链表

按照线性表的长度申请空间,并且初始化线性表元素个数和表的总长度。

//  初始化线型数组链表 
//  需要修改线性表,传入线性表地址 
Status InitList(SqList *L)
{// 按照表的长度申请空间,并赋值给*L的elem (*L).elem = (ElemType *)malloc(LIST_INIT_SIZE*sizeof(ElemType));// 申请失败,直接结束程序 if(!((*L).elem)){printf("数据空间申请失败!\n"); exit(OVERFLOW);}(*L).length = 0;(*L).listsize = LIST_INIT_SIZE;return OK;
}

1.3 插入数据

// 插入数据e在线性表的i位置上 1<=i<=n
//  需要修改线性表(分配空间已占满的时候需要扩容),传入线性表地址 

// 插入元素 1<=i<=n
//  需要修改线性表,传入线性表地址 
Status InsertList(SqList *L,int i,ElemType e) 
{ElemType *newbase,*p,*q; // 下标不合适返回失败 if(i<1 || i > (*L).length + 1)  return ERROR;// 当前分配的空间已占满,按增量重新分配 if((*L).length >= (*L).listsize) {newbase = (ElemType *)realloc((*L).elem,((*L).listsize + LISTINCREMENT) * sizeof(ElemType));if(!newbase){		printf("数据空间增量申请失败!\n"); exit(OVERFLOW);	}(*L).elem = newbase;(*L).listsize = (*L).listsize + LISTINCREMENT;}// 插入数据p = &(*L).elem[i-1];q = &(*L).elem[(*L).length+1];while(q > p){*q = *(q-1);q--; }*q = e;(*L).length++;return OK;
}

1.4 遍历列表

遍历列表过程中可能要做的事情不一样,因此将功能封装进函数中,然后将函数传入遍历函数中,进行执行。

// 链表的遍历 
Status ListTraverse(SqList L,void (*v)(ElemType *))
{int i;for(i=0;i<L.length;i++){v(L.elem+i);} printf("\n");return OK;
}

1.6 删除数据

// 删除元素 1<=i<=n
//  需要修改线性表,传入线性表地址 
// 通过e返回要删除的数值 
Status ListDelete(SqList *L,int i,ElemType *e) 
{	ElemType *p = NULL,*q=NULL; // 下标不合适返回失败 if(i<1 || i > (*L).length)  return ERROR;// 定位到要删除的位置p = &(*L).elem[i-1];*e = *p;q = &(*L).elem[(*L).length-1];// 将p指向空间,后面的数据向前移动。while(p <= q){*p = *(p+1);p++;} // 改变长度(*L).length--;return OK;	
}

1.7 销毁线性表

// 销毁线性表
//  需要修改线性表,传入线性表地址 
// 释放动态申请空间即可。
Status DestroyList(SqList *L) 
{// 释放堆区空间 free((*L).elem);// 将指针指向NULL,避免野指针。 (*L).elem = NULL;// 将线性表的长度的尺寸清0 (*L).length = 0;(*L).listsize = 0;return OK;
}

1.8 清空线性表

// 清空线性表
//  需要修改线性表,传入线性表地址 
// 将所有空间清为0
Status ClearList(SqList *L) 
{// 将空间里面的数据清0 int i;for(i=0;i<(*L).listsize;i++){(*L).elem[i] = 0;}// 将长度清0;(*L).length = 0; return OK;
}

1.9 判断线性表是否为空

// 判断线性表是否为空, 空表返回TRUE,否则返回FALSE
//  不需要修改线性表,仅用于访问,传入线性表即可 
Status ListEmpty(SqList L) 
{return L.length == 0 ? TRUE : FALSE; 
}

1.10 获取线性表的长度

// 获取 线性表长度
//  不需要修改线性表,仅用于访问,传入线性表即可 
Status ListLength(SqList L) 
{return L.length;
}

1.11 获取某个位置的数据

// 获取某个下标对应的元素 1<=i<=n
//  不需要修改线性表,仅用于访问,传入线性表即可 
Status GetElem(SqList L,int i,ElemType *e) 
{// 下标不合适返回失败 if(i<1 || i > L.length)  return ERROR;*e = L.elem[i-1];return OK; 
}

1.12 按照条件查找某个元素所在位置

将条件判断关系封装在函数内部,通过函数指针的形式传入到查找函数内。

// 查找元素e所在的位序, 1<=i<=n;如果找不到,返回0 
Status LocateElem(SqList L,ElemType e,Status (*compare)(ElemType,ElemType))
{int i;for(i=0;i<L.length;i++){if(compare(L.elem[i],e)){return i+1;} } return 0;
}

1.13 查找元素的前驱元素

// 查找元素的前驱
// 不是第一个元素有前驱 ,如果是第一个或者找不到则没有意义 
Status  PriorElem(SqList L,ElemType cur_e,ElemType *prev_e) 
{// idx为位序  1<=i<=n;int idx = LocateElem(L,cur_e,equal);if(idx <= 1) {return INFEASIBLE;}else{*prev_e = L.elem[idx-2];return OK;}	
}

1.14 查找元素的后驱元素

// 查找元素的后继 
Status  NextElem(SqList L,ElemType cur_e,ElemType *next_e) 
{// idx为位序  1<=i<=n;int idx = LocateElem(L,cur_e,equal);if(idx >= L.length) {		return INFEASIBLE;}else{*next_e = L.elem[idx];return OK;}	
}

1.15 两个线性表的并集

// 求两个线性表的并集
void unionList(SqList *La,SqList Lb) 
{int la_len = ListLength(*La);int lb_len = ListLength(Lb);int i;ElemType e;for(i=1;i<=lb_len;i++){// 将数据从Lb中读出放入到e里面去 GetElem(Lb,i,&e);// 检测e数据在不在La中。if(!LocateElem(*La,e,equal)) {// 不在将e放入到La中InsertList(La,++la_len,e); }}
}

1.16 两个有序线性表的合并

// 两个递增有序链表的合并。
void MergeList(SqList La,SqList Lb,SqList *Lc) 
{// 初始化LcInitList(Lc);int la_len = ListLength(La);int lb_len = ListLength(Lb); int i=1,j=1,k=0;ElemType ai,bj;while((i<=La.length) && (j <= Lb.length)){GetElem(La,i,&ai);GetElem(Lb,j,&bj);if(ai <= bj){InsertList(Lc,++k,ai);++i;}else{InsertList(Lc,++k,bj);++j;}}while(i <= La.length){GetElem(La,i++,&ai);InsertList(Lc,++k,ai);}while(j <= Lb.length){GetElem(Lb,j++,&bj);InsertList(Lc,++k,bj);}
}

1.17 判断两个数是否相等-查找辅助函数

// 辅助函数 -- 两个数据的相等判断 
Status equal(ElemType e1,ElemType e2)
{return e1 == e2 ? TRUE : FALSE;
}

1.18 访问数据--遍历的辅助函数

// 遍历辅助函数
void visit(ElemType *e)
{printf("%d ",*e);
} 

1.19 测试功能代码

void line_test(void)
{SqList l;InitList(&l);srand((unsigned int)time(NULL));	// 测试插入 int i;for(i=1;i<=5;i++){InsertList(&l,i,rand() % 10);} l.length = 5; printf("原始数据如下:");ListTraverse(l,visit);	// 接受删除的数据 int res;// 删除位置 int idx = rand() % 5 + 1; printf("删除位置:%d\n",idx);ListDelete(&l,idx,&res);printf("删除的数字是:%d,删除后:",res);ListTraverse(l,visit);//	ClearList(&l);
//	for(i=0;i<10;i++)
//	{
//		printf("%d ",l.elem[i]);
//	} 
//	printf("\n");// 线性表为空和 线性表长度测试if(ListEmpty(l)){printf("当前链表为空\n");} else{printf("链表当前的长度为:%d\n",ListLength(l));}// 获取某个元素测试int val;GetElem(l,1,&val);printf("第1个元素是:%d\n",val); // 查找元素的位序 测试int index = LocateElem(l,5,equal);printf("查找到5在数组的位序为:%d\n",index); // 查找元素的前驱测试Status res1;res1 = PriorElem(l,l.elem[0],&val);if(res1 != INFEASIBLE){printf("l.elem[3]的前驱是:%d\n",val);}// 查找元素的后继测试res1 = NextElem(l,l.elem[0],&val);if(res1 != INFEASIBLE){printf("l.elem[3]的后继是:%d\n",val);}// 两个线性表合并测试SqList l1,l2;InitList(&l1); InitList(&l2);int j;for(j=0;j<5;j++){l1.elem[j] = rand() % 10;} l1.length = 5; printf("线性表1:"); ListTraverse(l1,visit);for(j=0;j<3;j++){l2.elem[j] = rand() % 10;} l2.length = 3; printf("线性表2:");ListTraverse(l2,visit);unionList(&l1,l2);printf("合并后,线性表1:"); ListTraverse(l1,visit);// 两个增序线性表合并检测SqList l3,l4,l5;InitList(&l3); InitList(&l4);	InitList(&l5);	l3.elem[0] = 3;l3.elem[1] = 5;l3.elem[2] = 8;l3.elem[3] = 11;l3.length = 4;printf("线性表l3为:");ListTraverse(l3,visit);l4.elem[0] = 2;l4.elem[1] = 6;l4.elem[2] = 8;l4.elem[3] = 9;l4.elem[4] = 11;l4.elem[5] = 15;l4.elem[6] = 20;l4.length = 7;printf("线性表l4为:");ListTraverse(l4,visit);MergeList(l3,l4,&l5);printf("合并有序线性表l3和l4后为:");ListTraverse(l5,visit);
} 

 项目仓库地址

https://gitee.com/amyliyanice/data_struct.git


http://www.ppmy.cn/news/1192876.html

相关文章

vue详细安装教程

这里写目录标题 一、下载和安装node二、创建全局安装目录和缓存日志目录三、安装vue四、创建一个应用程序五、3x版本创建六、创建一个案例 一、下载和安装node 官网下载地址&#xff1a;https://nodejs.org/en/download 选择适合自己的版本&#xff0c;推荐LTS&#xff0c;长久…

Modbus转Profinet网关应用在自动上料机博图配置案例

该案例中的自动上料机通过使用Modbus转Profinet网关实现了与1200PLC和G120变频器的通信。这种通信方式能够实现设备之间的数据交换和控制命令传输&#xff0c;大大提升了自动上料机的运行效率和精度。通过使用该网关&#xff0c;1200PLC可以准确地向G120变频器发送控制命令&…

c++实现建造者模式

代码 director.h #pragma once #include<vector> #include<string> #include<iostream>class Product { private:std::vector<std::string> _parts;public:void Add(const std::string& part) {_parts.push_back(part);}void show(){std::cout &…

03-Vue中的常用指令的使用,事件及其修饰符

常用指令 指令语法和插值语法 Vue框架中的所有指令的名字都以v-开始,完整语法格式<HTML标签 v-指令名:参数"javascript表达式(表达式的结果是一个值)"></HTML标签>: 指令的职责是当表达式的值改变时,将其产生的连带影响,响应式地作用于DOM元素不是所有…

【进程控制⑥】:进程替换/exec*()系列接口

【进程控制⑥】&#xff1a;进程替换/ exec*(&#xff09;系列接口 一.进程替换原理二.替换特点1.独立性2.唯一性3.不变性4.不返回 三.程序替换应用【exec*系列系统调用】①execl&#xff1a;②execlp&#xff1a;③execv&#xff1a;④execle&#xff1a; 一.进程替换原理 我…

鲁班猫4(RK3588S)配置Realsense SDK和Realsense ROS(D435i T265)

0 环境 鲁班猫4开发板&#xff08;RK3588S&#xff09;Ubuntu 20.04 &#xff08;lubancat-rk3588-ubuntu20.04-gnome-20230829_update&#xff09;D435iT265ROS noeticRealsense SDK v2.53.1Realsense ROS v2.3.2 1 安装ROS 建议使用fishros.com的ROS一键安装 wget http:/…

msvcp120.dll下载修复详细方法与解决方法

MSVCP120.dll文件丢失是一个常见的问题&#xff0c;它通常会导致某些程序无法正常运行。这个问题可能是由于多种原因引起的&#xff0c;例如系统更新、软件卸载或病毒感染等。在这篇文章中&#xff0c;我将为您提供四种解决MSVCP120.dll文件丢失的方法&#xff0c;帮助您快速恢…

Asterisk Ubuntu 安装

更新环境 sudo apt update sudo apt install wget build-essential git autoconf subversion pkg-config libtool sudo contrib/scripts/get_mp3_source.sh A addons/mp3 A addons/mp3/common.c A addons/mp3/huffman.h A addons/mp3/tabinit.c A addons/mp3/Ma…