当前位置: 首页 > news >正文

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

本栏目致力于从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

相关文章:

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

本栏目致力于从0开始使用纯C语言将经典算法转换成能够直接上机运行的程序&#xff0c;以项目的形式详细描述数据存储结构、算法实现和程序运行过程。 参考书目如下&#xff1a; 《数据结构C语言版-严蔚敏》 《数据结构算法解析第2版-高一凡》 软件工具&#xff1a; dev-cpp 0…...

pthread 名字设置及线程标识符获取

pthread 名字设置及ID获取 pthread_setname_np 函数原型&#xff1a; int pthread_setname_np(pthread_t thread, const char *name);thread&#xff1a;要设置名称的线程标识符&#xff08;pthread_t&#xff09;。name&#xff1a;要设置的线程名称&#xff08;以字符串形式…...

17、Flink 之Table API: Table API 支持的操作(1)

Flink 系列文章 1、Flink 部署、概念介绍、source、transformation、sink使用示例、四大基石介绍和示例等系列综合文章链接 13、Flink 的table api与sql的基本概念、通用api介绍及入门示例 14、Flink 的table api与sql之数据类型: 内置数据类型以及它们的属性 15、Flink 的ta…...

Ubuntu:解决PyCharm中不能输入中文或者输入一个中文解决方法

1.问题&#xff1a; Ubuntu22.04中&#xff0c;在pycharm里打字输入中文&#xff0c;每次都是只能输入第一个中文&#xff0c;后面输入的都变成了英文字母。。。无论咋调输入法&#xff0c;都没用&#xff0c;反正除了第一个字其他的输进去都是英文&#xff0c;而且汉字下面还…...

Vue3.0 reactive与ref :VCA模式

简介 Vue3 最大的一个变动应该就是推出了 CompositionAPI&#xff0c;可以说它受ReactHook 启发而来&#xff1b;它我们编写逻辑更灵活&#xff0c;便于提取公共逻辑&#xff0c;代码的复用率得到了提高&#xff0c;也不用再使用 mixin 担心命名冲突的问题。 ref 与 reactive…...

项目实战 | 使用Linux宝塔面板搭建商城公众号小程序基础框架

项目实战 | 使用Linux宝塔面板搭建商城公众号&小程序基础框架 1. 小程序/公众号运行的必备条件2. 准备阿里云ECS主机3. 宝塔面板基本配置4. 通过宝塔面板安装相关服务5. 新建站点并进行初始配置6. 服务配置6.1. PHP配置6.2. 数据库配置6.3. Redis配置6.4. 消息队列Supervis…...

IDEA远程调试代码

IDEA->RUN->Edit Configurations 端口随便选一个&#xff0c;选择调试模块&#xff0c;然后用IDEA生成的命令调试 java -agentlib:jdwptransportdt_socket,servery,suspendn,address*:8081 -jar backend-1.18.11.jar &...

目标检测 图像处理 计算机视觉 工业视觉

目标检测 图像处理 计算机视觉 工业视觉 工业表盘自动识别&#xff08;指针型和数值型&#xff09;智能水尺识别电梯中电动车识别&#xff0c;人数统计缺陷检测&#xff08;半导体&#xff0c;电子元器件等&#xff09;没带头盔检测基于dlib的人脸识别抽烟检测和睡岗检测/驾驶疲…...

【1day】宏景OA get_org_tree.jsp接口SQL注入漏洞学习

注:该文章来自作者日常学习笔记,请勿利用文章内的相关技术从事非法测试,如因此产生的一切不良后果与作者无关。 目录...

设计模式-迭代子模式

迭代子模式是一种行为设计模式&#xff0c;它提供了一种访问和遍历聚合对象中各个元素的方法&#xff0c;而不需要暴露聚合对象的内部表示。迭代子模式将遍历聚合对象的责任交给了迭代子对象&#xff0c;从而实现了聚合对象和迭代子对象的解耦。 在Java中&#xff0c;迭…...

绿色通道 快速理赔,渤海财险用实干书写服务品牌

7月底&#xff0c;受台风“杜苏芮”影响&#xff0c;北京市连续强降雨&#xff0c;西部、西南部、南部遭遇特大暴雨&#xff0c;房山、门头沟、丰台等地陆续出现山洪暴发现象。      灾害无情人有情&#xff0c;为更好地保障人民群众生命财产安全&#xff0c;渤海财险北京分…...

微信小程序怎么制作?【小程序开发平台教学】

随着移动互联网的快速发展&#xff0c;微信小程序已经成为了人们日常生活中不可或缺的一部分。从购物、支付、出行到社交、娱乐、教育&#xff0c;小程序几乎涵盖了我们生活的方方面面。那么&#xff0c;对于有营销需求的企业商家来说&#xff0c;如何制作一个自己的微信小程序…...

HTML、CSS和JavaScript,实现换肤效果的原理

这篇涉及到HTML DOM的节点类型、节点层级关系、DOM对象的继承关系、操作DOM节点和HTML元素 还用到HTML5的本地存储技术。 换肤效果的原理&#xff1a;是在选择某种皮肤样式之后&#xff0c;通过JavaScript脚本来加载选中的样式&#xff0c;再通过localStorage存储。 先来回忆…...

2103. 环和杆

2103. 环和杆 难度: 简单 来源: 每日一题 2023.11.02 总计有 n 个环&#xff0c;环的颜色可以是红、绿、蓝中的一种。这些环分别穿在 10 根编号为 0 到 9 的杆上。 给你一个长度为 2n 的字符串 rings &#xff0c;表示这 n 个环在杆上的分布。rings 中每两个字符形成一个…...

YOLOv5:修改backbone为SPPCSPC

YOLOv5&#xff1a;修改backbone为SPPCSPC 前言前提条件相关介绍SPPCSPCYOLOv5修改backbone为SPPCSPC修改common.py修改yolo.py修改yolov5.yaml配置 参考 前言 记录在YOLOv5修改backbone操作&#xff0c;方便自己查阅。由于本人水平有限&#xff0c;难免出现错漏&#xff0c;敬…...

css中flexbox和grid的区别

css中flexbox和grid的区别 我们是不是被那些不会按预期排列的元素所影响&#xff1f;这篇文章我们将深入探讨css中flexbox和grid的布局。通过了解他们的主要差异&#xff0c;我们会发现这些布局是如何改变我们网站的风格。 理解CSS布局 css布局是网页设计的一个重要方面&…...

uniapp循环对象列表---点击列表切换选中不同状态

目录 源码图片最后 源码 <template><view><ul><li v-for"(item, index) in list" click"toggleSelection(index)" :class"{selected: selectedIndex index}">{{ item }}<view :class"{selected: selectedInde…...

【使用Python编写游戏辅助工具】第二篇:键盘监听的应用

前言 这里是【使用Python编写游戏辅助工具】的第二篇&#xff1a;键盘监听的应用。本文主要介绍使用Python实现事件监听功能。 键盘监听是指通过编程的方式监控用户在键盘上的按键操作。 在这里键盘监听的主要用途是&#xff1a; 监听我们按下的按键&#xff0c;如果按下了指…...

Shiny Server和ShinyProxy是什么,有什么区别?

调研以及参与过多个生物公司的生信工具研发&#xff0c;不管是ShinyServer还是ShinyProxy都有一定研究&#xff0c;尤其是ShinyServer。如果仅是本地化测试想快速的搭建Shiny应用&#xff0c;我推荐用Shiny Server&#xff0c;如果多并发用户且更好的线上管理Shiny应用&#xf…...

Java 客户端、服务端NIO大文件传输

一、需求 公司电脑不让使用U盘&#xff0c;又不想通过公司聊天软件传输&#xff0c;怕被监控。但是通过QQ、微信传输文件对文件大小又有限制。基于种种原因&#xff0c;自己简单写了个服务端、客户端进行文件传输&#xff0c;大文件最好在局域网内进行数据传输。 二、pom依赖…...

Chapter03-Authentication vulnerabilities

文章目录 1. 身份验证简介1.1 What is authentication1.2 difference between authentication and authorization1.3 身份验证机制失效的原因1.4 身份验证机制失效的影响 2. 基于登录功能的漏洞2.1 密码爆破2.2 用户名枚举2.3 有缺陷的暴力破解防护2.3.1 如果用户登录尝试失败次…...

Java 语言特性(面试系列1)

一、面向对象编程 1. 封装&#xff08;Encapsulation&#xff09; 定义&#xff1a;将数据&#xff08;属性&#xff09;和操作数据的方法绑定在一起&#xff0c;通过访问控制符&#xff08;private、protected、public&#xff09;隐藏内部实现细节。示例&#xff1a; public …...

Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以?

Golang 面试经典题&#xff1a;map 的 key 可以是什么类型&#xff1f;哪些不可以&#xff1f; 在 Golang 的面试中&#xff0c;map 类型的使用是一个常见的考点&#xff0c;其中对 key 类型的合法性 是一道常被提及的基础却很容易被忽视的问题。本文将带你深入理解 Golang 中…...

Python:操作 Excel 折叠

💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 Python 操作 Excel 系列 读取单元格数据按行写入设置行高和列宽自动调整行高和列宽水平…...

前端倒计时误差!

提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...

23-Oracle 23 ai 区块链表(Blockchain Table)

小伙伴有没有在金融强合规的领域中遇见&#xff0c;必须要保持数据不可变&#xff0c;管理员都无法修改和留痕的要求。比如医疗的电子病历中&#xff0c;影像检查检验结果不可篡改行的&#xff0c;药品追溯过程中数据只可插入无法删除的特性需求&#xff1b;登录日志、修改日志…...

【项目实战】通过多模态+LangGraph实现PPT生成助手

PPT自动生成系统 基于LangGraph的PPT自动生成系统&#xff0c;可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析&#xff1a;自动解析Markdown文档结构PPT模板分析&#xff1a;分析PPT模板的布局和风格智能布局决策&#xff1a;匹配内容与合适的PPT布局自动…...

Qt Http Server模块功能及架构

Qt Http Server 是 Qt 6.0 中引入的一个新模块&#xff0c;它提供了一个轻量级的 HTTP 服务器实现&#xff0c;主要用于构建基于 HTTP 的应用程序和服务。 功能介绍&#xff1a; 主要功能 HTTP服务器功能&#xff1a; 支持 HTTP/1.1 协议 简单的请求/响应处理模型 支持 GET…...

python爬虫:Newspaper3k 的详细使用(好用的新闻网站文章抓取和解析的Python库)

更多内容请见: 爬虫和逆向教程-专栏介绍和目录 文章目录 一、Newspaper3k 概述1.1 Newspaper3k 介绍1.2 主要功能1.3 典型应用场景1.4 安装二、基本用法2.2 提取单篇文章的内容2.2 处理多篇文档三、高级选项3.1 自定义配置3.2 分析文章情感四、实战案例4.1 构建新闻摘要聚合器…...

相机从app启动流程

一、流程框架图 二、具体流程分析 1、得到cameralist和对应的静态信息 目录如下: 重点代码分析: 启动相机前,先要通过getCameraIdList获取camera的个数以及id,然后可以通过getCameraCharacteristics获取对应id camera的capabilities(静态信息)进行一些openCamera前的…...