热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

【2.1】顺序表

第二章序言上一章的两节介绍了常用的算法,这些算法用来处理零散的数据。实际上我们有时候处理的数据之间是存在一种或者多种特定的关系时,我们称这些关系为结构

第二章序言

上一章的两节介绍了常用的算法,这些算法用来处理零散的数据。实际上我们有时候处理的数据之间是存在一种或者多种特定的关系时,我们称这些关系为结构。通常数据之间有三种基本的机构。

(1)线性结构:数据元素之间为一对一的关系。

(2)树形结构:数据元素之间为一对多的关系。

(3)网状结构:数据元素之间为多对多的关系。


什么是线性表?

线性表示最基本、最简单、也是最常用的一种数据结构。它是一个含有n个节点的有限序列,不同线性表的数据元素可以不痛,但是对于同一个线性表,各个数据元素必须有相同的数据类型,即同一线性表中各个数据元素具有相同的数据类型,每个数据元素的长度相同。

线性表的数据结构具有以下特征:

a、有且只有一个首元素

b、有且只有一个末元素

c、除末元素之外,其余元素均有唯一的后继元素

d、除首元素之外,其余元素均有唯一的前驱元素

顺序表是用一组地址连续的存储单元依次保存线性表中的数据元素,由于它使用这种存储结构,使得程序中可以很容易的访问顺序表中的任何元素,也可以方便的将数据元素添加到顺序表的末尾。

顺序表有什么结构特征?

1、采用一组地址连续的存储单元来存储数据

2、表的存储容量长度不易改变

3、可以随机查找

4、可频繁访问元素

5、不适用于元素的频繁增删操作


项目代码案例,项目结构如下:

文件1:Utils.h

#ifndef _YELER_082_02
#define _YELER_082_02
typedef int ElemType;
typedef int Status;
typedef void MyVoid;
typedef void MyVoid;
typedef int ElemType;
typedef int Status;
typedef struct
{ElemType * elem; //存储空间基址int length; //当前实际使用长度int listsize; //当前分配存储容量
}SqList;
//比较函数的声明
Status compare(ElemType e1, ElemType e2);MyVoid visit(ElemType e, ElemType i);#endif
文件二:method.h

#ifndef _YELER_082_01
#define _YELER_082_01
#include "Utils.h"
Status Init(SqList &L);//构造一个空的线性表Status Destory(SqList &L);//销毁线性表LStatus ClearList(SqList &L);//将L表重置为空表Status ListEmpty(SqList L);//判断L表是否为空Status ListLength(SqList L);//返回L表中数据元素的长度Status ListInsert(SqList &L, ElemType i, ElemType e);//向表中的第i个位置插入元素eStatus GetElem(SqList L,ElemType i,ElemType &e);//取出L表中指定位置的元素Status LocateElem(SqList L,ElemType e, Status(*compare)(ElemType, ElemType));//找出在L表中第一个与e匹配的元素Status PriorElem(SqList L,ElemType cur_e,ElemType &pre_e);//找出当前元素cur_e的前驱pre_eStatus NextElem(SqList L,ElemType cur_e,ElemType &next_e);//找出当前元素cur_e的后继next_eStatus ListDelete(SqList &L,ElemType i,ElemType &e);//删除表中第i个位置的元素eStatus ListTraverse(SqList L, MyVoid(*visit)(ElemType,ElemType));//遍历表中的每个元素在visit函数中执行一遍Status ListSort(SqList &L);//对链表中的数据排序,从小到大MyVoid MergeList_Sq(SqList La, SqList Lb, SqList &Lc); //已知线性表La和Lb,归并La和Lb的到线性表Lc,Lc的元素按照非递减排列
#endif
文件3:Utils.cpp

#include
typedef int ElemType;
typedef int Status;typedef void MyVoid;//比较的工具函数
Status compare(ElemType e1, ElemType e2)
{if (e1 == e2)return true;//在C++中有bool类型的定义,C语言中是没有的return false;}//输出的工具函数
MyVoid visit(ElemType e,ElemType i)
{printf("第%d个元素是%d\n", i, e);
}
文件4:method.cpp

#include
#include
#include "Utils.h"#define LIST_INIT_SIZE 100 //线性表存储空间的初始分配量
#define LISTINCREMENT 10 //线性表存储空间的分配增量
#define OK 1
#define TRUE 1
#define FALSE 0
#define NOTEXIST 0
#define OVERFLOW -1
#define ERROR -2//构造一个空的线性表
Status Init(SqList &L)
{L.elem = (ElemType *)malloc(LIST_INIT_SIZE*sizeof(ElemType));//给线性表分配初始空间400个字节if (!L.elem){exit(OVERFLOW);}L.length = 0;//指定实际长度为0,存的数据个数L.listsize = LIST_INIT_SIZE;//指定分配空间长度return OK;
}//销毁线性表L
Status Destory(SqList &L)
{if (!L.elem)//如果线性表为空,返回错误码{return ERROR;}free(L.elem);return OK;
}//将L表重置为空表
Status ClearList(SqList &L)
{if (!L.elem){return ERROR;}L.length = 0;return OK;
}//判断L表是否为空
Status ListEmpty(SqList L)
{if (!L.elem){return ERROR;}if (L.length!=0){return TRUE;}return FALSE;
}//返回L表中数据元素的长度
Status ListLength(SqList L)
{if (!L.elem){return ERROR;}return L.length;
}//取出L表中指定位置的元素
Status GetElem(SqList L, ElemType i, ElemType &e)
{if (!L.elem){return ERROR;}if (i<1 || i>L.length &#43; 1){return ERROR;}e&#61; L.elem[i-1];return OK;
}//找出在L表中第一个与e匹配的元素,第三个参数是指向比较函数的指针
Status LocateElem(SqList L, ElemType e, Status(*compare)(ElemType, ElemType))
{//在顺序线性表L中查找第1个值与e满足compare()的元素的为序//若找到&#xff0c;则返回其在L中的位序&#xff0c;否则返回0int i &#61; 1;//i的初值为第1个元素的位序ElemType *p &#61; L.elem;//定义一个指针指向线性表的起始位置while (i<&#61;L.length&&!(*compare)(*p&#43;&#43;,e))//线性表的数据在长度范围内没有找到相同的数值&#xff0c;位置以此后移{i&#43;&#43;;}if (i<&#61;L.length){return i;//返回的是在第几个位置的元素}else{return NOTEXIST;}
}//找出当前元素cur_e的前驱pre_e
Status PriorElem(SqList L, ElemType cur_e, ElemType &pre_e)
{ElemType *p &#61; NULL,i&#61;1;p &#61; L.elem;//定义一个指针指向头结点if (!L.elem){return ERROR;}if (*p&#61;&#61;cur_e)//如果当前被找的参照物为第一个节点&#xff0c;那么返回错误{return ERROR;}//遍历找到当前的节点所在的位置while (i<&#61;L.length){if (*p&#61;&#61;cur_e){pre_e &#61; *(p - 1); break;}i&#43;&#43;;p&#43;&#43;;}if (i <&#61; L.length){return OK;}else{return NOTEXIST;}
}//找出当前元素cur_e的后继next_e
Status NextElem(SqList L, ElemType cur_e, ElemType &next_e)
{ElemType *p &#61; NULL,i&#61;0;p &#61; L.elem;//当传递过来的线性表为空&#xff0c;或者ElemType cur_e为最后的一个节点的时候&#xff0c;返回错误值if (!p||cur_e&#61;&#61;L.elem[L.length-1])//数组表示是从零开始&#xff0c;L.elem[L.length-1]代表线性表中的最后一个元素{return ERROR;}while (i<&#61;L.length){if (cur_e&#61;&#61;*p){next_e &#61; *(p &#43; 1); break;}i&#43;&#43;;p&#43;&#43;;}if (i <&#61; L.length){return OK;}else{return NOTEXIST;}
}//向表中的第i个位置插入元素e
Status ListInsert(SqList &L, ElemType i, ElemType e)
{ElemType *newbase &#61; NULL, *q &#61; NULL, *p &#61; NULL;if (!L.elem){return ERROR;}if (i<1 || i>L.length &#43; 1){return ERROR;}if (L.length >&#61; L.listsize){newbase &#61; (ElemType*)realloc(L.elem, (L.listsize &#43; LISTINCREMENT)*sizeof(ElemType));if (!newbase){exit(OVERFLOW);}L.elem &#61; newbase;//新基址L.listsize &#43;&#61; LISTINCREMENT;}q &#61; &(L.elem[i - 1]);//得到要插入元素的地址//插入之后的元素位置后移for (p &#61; &(L.elem[L.length - 1]); p >&#61; q; --p){*(p &#43; 1) &#61; *p;}*q &#61; e;&#43;&#43;L.length;return OK;
}//删除表中第i个位置的元素e
Status ListDelete(SqList &L, ElemType i, ElemType &e)
{ElemType *p &#61; NULL,*q&#61; NULL,t&#61;1;p &#61; L.elem;if (!L.elem){return ERROR;}if (i<1||i>L.length&#43;1){return ERROR;}//删除涉及到移位操作p &#61; &L.elem[i];//获取需要删除的元素的地址e &#61; *p;//将被删除的数据放置到变量中&#xff0c;不能放到下方&#xff0c;因为指针的位置变了q &#61; L.elem &#43; L.length - 1;//q是指向表尾节点的指针for (&#43;&#43;p; p <&#61;q ; p&#43;&#43;){*(p - 1) &#61; *(p);}L.length--;return OK;
}//遍历表中的每个元素在visit函数中执行一遍typedef void MyVoid;
Status ListTraverse(SqList L, MyVoid(*visit)(ElemType, ElemType))
{ElemType *p &#61; NULL,i;p &#61; L.elem;for (i &#61; 1; i <&#61;L.length; i&#43;&#43;,p&#43;&#43;){(*visit)(*p,i);}return OK;
}//对链表中的数据排序&#xff0c;从小到大&#xff0c;采用冒泡排序法
Status ListSort(SqList &L)
{ElemType *p &#61; NULL, i, j;ElemType temp;for ( i &#61; 0; i L.elem[j&#43;1]){temp &#61; L.elem[j];L.elem[j ] &#61; L.elem[j&#43;1];L.elem[j&#43;1] &#61; temp;}}}return OK;
}//已知线性表La和Lb,归并La和Lb的到线性表Lc&#xff0c;Lc的元素按照非递减排列
MyVoid MergeList_Sq(SqList La, SqList Lb, SqList &Lc)
{ElemType *pa &#61; NULL, *pb &#61; NULL, *pc &#61; NULL;ElemType *pa_last &#61; NULL, *pb_last &#61; NULL;pa &#61; La.elem, pb &#61; Lb.elem;Lc.listsize &#61; Lc.length &#61; La.length &#43; Lb.length;pc &#61; Lc.elem &#61; (ElemType*)malloc(Lc.listsize*sizeof(ElemType));//为pc申请空间if (!Lc.elem) exit(OVERFLOW);//存储分配失败pa_last &#61; La.elem &#43; La.length - 1;//获取线性表La的最后一个元素的指针pb_last &#61; Lb.elem &#43; Lb.length - 1;//获取线性表Lb的最后一个元素的指针while (pa <&#61; pa_last&&pb <&#61; pb_last){if (*pa <&#61; *pb){*pc&#43;&#43; &#61; *pa&#43;&#43;;//*pc &#61; *pa; pc &#61; pc&#43;1; pa &#61; pa&#43;1;}else{*pc&#43;&#43; &#61; *pb&#43;&#43;;}}while (pa<&#61;pa_last){*pc&#43;&#43; &#61; *pa&#43;&#43;;//插入La中的剩余元素}while (pb<&#61;pb_last){*pc&#43;&#43; &#61; *pb&#43;&#43;;//插入Lb中的剩余元素}
}
文件5&#xff1a;main.cpp

#include
#include
#include"method.h"
#include"Utils.h"
int main()
{SqList L,La,Lb,Lc;ElemType i &#61; 0,e&#61;0,*p&#61;NULL,flag;printf("*************线性表的顺序操作**********\n");printf("初始化顺序表:");i&#61;Init(L);printf("%d\n",i);/*printf("————————————————————\n");printf("销毁顺序表:");i &#61; Destory(L);printf("%d\n", i);*//*printf("————————————————————\n");printf("将顺序表置空:");i&#61;ClearList(L);printf("%d\n",i);*//*printf("————————————————————\n");printf("判断顺序表是否为空:");i &#61; ListEmpty(L);printf("%d\n", i);*//*printf("————————————————————\n");printf("输出顺序表的长度:");i&#61;ListLength(L);printf("%d\n", i);*/printf("————————————————————\n");printf("向表中插入元素\n");ListInsert(L, 1, 5);ListInsert(L, 2, 3);ListInsert(L, 1, 1);ListInsert(L, 2, 7);ListInsert(L, 2, 10);printf("插入元素为:");p &#61; L.elem;//这里另外找一个指针专门用来遍历&#xff0c;不要轻易的移动L.elem&#xff0c;否则会对后面的操作造成麻烦for ( i &#61; 0; i %d\n",i);}else{printf("取出L表中7的前驱元素不存在!\n");}printf("————————————————————\n");printf("取出L表中7的后继元素:");flag &#61; NextElem(L, 7, i);if (flag){printf("L表中7的后继元素是-->%d\n", i);}else{printf("取出L表中7的后继元素不存在!\n");}printf("————————————————————\n");printf("删除L表中数值为7元素:\n");ListDelete(L, 3, i);printf("L表中被删除的元素是%d\n", i);printf("————————————————————\n");printf("遍历L表中的元素并打印:\n");ListSort(L);//调用排序函数MyVoid(*Visit)(ElemType, ElemType);//定义一个指向返回空值&#xff0c;含有两个参数&#xff0c;名称为Visit的函数指针Visit &#61; visit;//将函数指针指向一个函数ListTraverse(L, Visit);printf("————————————————————\n");Init(La), Init(Lb);//初始化两个线性表//向线性表中插入数据ListInsert(La, 1, 5);ListInsert(La, 2, 9);ListInsert(La, 3, 4);ListInsert(Lb, 1, 2);ListInsert(Lb, 1, 7);ListInsert(Lb, 1, 1);ListInsert(Lb, 2, 8);ListSort(La), ListSort(Lb);//对线性表进行排序MergeList_Sq(La, Lb, Lc);ListTraverse(Lc, Visit);return 0;
}
运行截图&#xff1a;




推荐阅读
  • 在尝试对 QQmlPropertyMap 类进行测试驱动开发时,发现其派生类中无法正常调用槽函数或 Q_INVOKABLE 方法。这可能是由于 QQmlPropertyMap 的内部实现机制导致的,需要进一步研究以找到解决方案。 ... [详细]
  • 如何在Java中使用DButils类
    这期内容当中小编将会给大家带来有关如何在Java中使用DButils类,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。D ... [详细]
  • Ihavetwomethodsofgeneratingmdistinctrandomnumbersintherange[0..n-1]我有两种方法在范围[0.n-1]中生 ... [详细]
  • 字符串学习时间:1.5W(“W”周,下同)知识点checkliststrlen()函数的返回值是什么类型的?字 ... [详细]
  • 字节流(InputStream和OutputStream),字节流读写文件,字节流的缓冲区,字节缓冲流
    字节流抽象类InputStream和OutputStream是字节流的顶级父类所有的字节输入流都继承自InputStream,所有的输出流都继承子OutputStreamInput ... [详细]
  • 本文详细介绍了 PHP 中对象的生命周期、内存管理和魔术方法的使用,包括对象的自动销毁、析构函数的作用以及各种魔术方法的具体应用场景。 ... [详细]
  • 在软件开发过程中,经常需要将多个项目或模块进行集成和调试,尤其是当项目依赖于第三方开源库(如Cordova、CocoaPods)时。本文介绍了如何在Xcode中高效地进行多项目联合调试,分享了一些实用的技巧和最佳实践,帮助开发者解决常见的调试难题,提高开发效率。 ... [详细]
  • 题目《BZOJ2654: Tree》的时间限制为30秒,内存限制为512MB。该问题通过结合二分查找和Kruskal算法,提供了一种高效的优化解决方案。具体而言,利用二分查找缩小解的范围,再通过Kruskal算法构建最小生成树,从而在复杂度上实现了显著的优化。此方法不仅提高了算法的效率,还确保了在大规模数据集上的稳定性能。 ... [详细]
  • 本文提出了一种基于栈结构的高效四则运算表达式求值方法。该方法能够处理包含加、减、乘、除运算符以及十进制整数和小括号的算术表达式。通过定义和实现栈的基本操作,如入栈、出栈和判空等,算法能够准确地解析并计算输入的表达式,最终输出其计算结果。此方法不仅提高了计算效率,还增强了对复杂表达式的处理能力。 ... [详细]
  • 你的问题在于:1. 代码格式混乱,缺乏必要的缩进,导致可读性极低;2. 使用 `strlen()` 和 `malloc()` 函数时,必须包含相应的头文件;3. `write()` 函数的返回值处理不当,建议检查并处理其返回值以确保程序的健壮性。此外,建议在编写代码时遵循良好的编程规范,增加代码的可维护性和可读性。 ... [详细]
  • C语言中全部可用的数学函数有哪些?2.longlabs(longn);求长整型数的绝对值。3.doublefabs(doublex);求实数的绝对值。4.doublefloor(d ... [详细]
  • DAO(Data Access Object)模式是一种用于抽象和封装所有对数据库或其他持久化机制访问的方法,它通过提供一个统一的接口来隐藏底层数据访问的复杂性。 ... [详细]
  • 本文介绍如何使用 Python 的 DOM 和 SAX 方法解析 XML 文件,并通过示例展示了如何动态创建数据库表和处理大量数据的实时插入。 ... [详细]
  • 深入解析C语言中结构体的内存对齐机制及其优化方法
    为了提高CPU访问效率,C语言中的结构体成员在内存中遵循特定的对齐规则。本文详细解析了这些对齐机制,并探讨了如何通过合理的布局和编译器选项来优化结构体的内存使用,从而提升程序性能。 ... [详细]
  • 作文记录:合并区间的技巧与应用
    本文详细记录了合并区间问题的解题技巧与应用场景。首先介绍了问题背景和题目描述,接着从排序最大值的角度探讨了解决思路,并提供了具体的程序代码及运行结果。此外,还探讨了其他可能的解决方案。最后,对整个解题过程进行了总结,为读者提供了全面的理解和参考。 ... [详细]
author-avatar
mobiledu2502871703
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有