数据结构4:顺序表(线性表的顺序存储结构)及C语言实现
逻辑结构上呈线性分布的数据元素在实际的物理存储结构中也同样相互之间紧挨着,这种存储结构称为线性表的顺序存储结构。
也就是说,逻辑上具有线性关系的数据按照前后的次序全部存储在一整块连续的内存空间中,之间不存在空隙,这样的存储结构称为顺序存储结构。
使用顺序存储结构存储的数据,第一个元素所在的地址就是这块存储空间的首地址。通过首地址,可以轻松访问到存储的所有的数据,只要首地址不丢,数据永远都能找着(一根绳上的蚂蚱,要有就都有)。
使用线性表的顺序存储结构生成的表,称为顺序表。

图1 顺序表结构示意图
顺序表的实现方法
顺序表中存放数据的特点和数组这种数据类型完全吻合,所以顺序表的实现使用的是数组。
顺序表的存储结构
在建立顺序表时,除了预先申请内存空间,还需要实时记录顺序表的长度和顺序表本身申请的内存大小,便于后期对顺序表中的数据元素进行调取。
所以,要自定义顺序表的结构:
typedef struct Table{
int * head;//声明了一个名为head的长度不确定的数组,也叫“动态数组”
int length;//记录当前顺序表的长度
int size;//记录顺序表分配的存储容量
}table;
顺序表的创建
顺序表的建立,也就是顺序表进行初始化,在预先申请内存空间的同时,给变量size和length赋初值:
table initTable()
{
table t;
t.head=(int*)malloc(Size*sizeof(int)); //构造一个空的顺序表,动态申请存储空间
if (!t.head) //如果申请失败,作出提示并直接退出程序
{
printf("初始化失败");
exit();
}
t.length=; //空表的长度初始化为0
t.size=Size; //空表的初始存储空间为Size
return t;
}
顺序表查找元素
在数组中查找某个数据元素时,可以采取多种查找算法,例如二分查找、插值查找、斐波那契查找算法等。
根据顺序表中存储的数据的特点,选择合适的算法。这里,采用顺序查找算法(普通的遍历算法)。
实现代码:
//查找函数,其中,elem表示要查找的数据元素的值
int selectTable(table t, int elem)
{
for (int i=; i<t.length; i++)
{
if (t.head[i] == elem)
{
return i+;
}
}
return -;//如果查找失败,返回-1
}
顺序表中更改元素
顺序表中更改数据元素,最简单直接的方式就是:调用查找算法找到该数据元素的位置,直接在该位置上更改。
实现代码:
//更改函数,其中,elem为要更改的元素,newElem为新的数据元素
table amendTable(table t, int elem, int newElem)
{
int add = selectTable(t, elem);
t.head[add-] = newElem; //由于返回的是元素在顺序表中的位置,所以-1就是该元素在数组中的下标
return t;
}
顺序表插入元素
插入数据元素,无非三种情况:
- 在表头插入
- 在表的中间某个位置插入
- 直接尾随顺序表,作为表的最后一个元素
无论在顺序表的什么位置插入数据元素,解决办法都是:找到要插入的位置,将后续数据元素整体向后移动一个位置,最后直接在腾出来的位置上插入数据元素。
实现代码:
//插入函数,其中,elem为插入的元素,add为插入到顺序表的位置
table addTable(table t, int elem, int add)
{
//判断插入本身是否存在问题(如果插入元素位置比整张表的长度+1还大(如果相等,是尾随的情况),
//或者插入的位置本身不存在,程序作为提示并自动退出)
if (add>t.length+1 || add<)
{
printf("插入位置有问题");
return t;
}
//做插入操作时,首先需要看顺序表是否有多余的存储空间提供给插入的元素,如果没有,需要申请
if (t.length == t.size)
{
t.head=(int *)realloc(t.head, (t.size+)*sizeof(int));
if (!t.head)
{
printf("存储分配失败");
return t;
}
t.size += ;
}
//插入操作,需要将从插入位置开始的后续元素,逐个后移
for (int i=t.length-; i>=add-; i--)
{
t.head[i+] = t.head[i];
}
//后移完成后,直接将所需插入元素,添加到顺序表的相应位置
t.head[add-] = elem;
//由于添加了元素,所以长度+1
t.length++;
return t;
}
注意:在此程序中,当数组存储空间不足时,使用realloc函数每次额外多申请 1 个int型的存储空间,这么做还不是最优。最好的办法就是每次发现空间不够时,多申请几个内存空间,这么做的好处是:在后续做插入操作过程中不需要每次都运行realloc函数,提高了程序的运行效率。
顺序表删除元素
在数组中删除元素时,只需将该元素所在位置后的所有数据元素整体前移 1 个位置即可。
实现代码:
table delTable(table t,int add)
{
if (add>t.length || add<)
{
printf("被删除元素的位置有误");
exit();
}
//删除操作
for (int i=add; i<t.length; i++)
{
t.head[i-]=t.head[i];
}
t.length--;
return t;
}
完整的程序
#include <stdio.h>
#include <stdlib.h>
#define Size 4
typedef struct Table
{
int * head;
int length;
int size;
}table;
table initTable()
{
table t;
t.head=(int*)malloc(Size*sizeof(int));
if (!t.head)
{
printf("初始化失败");
exit();
}
t.length = ;
t.size = Size;
return t;
}
table addTable(table t,int elem,int add)
{
if (add>t.length+1 || add<)
{
printf("插入位置有问题");
return t;
}
if (t.length >= t.size)
{
t.head = (int *)realloc(t.head, (t.size+)*sizeof(int));
if (!t.head)
{
printf("存储分配失败");
}
t.size += ;
}
for (int i=t.length-; i>=add-; i--)
{
t.head[i+]=t.head[i];
}
t.head[add-] = elem;
t.length++;
return t;
}
table delTable(table t,int add)
{
if (add>t.length || add<)
{
printf("被删除元素的位置有误");
exit();
}
for (int i=add; i<t.length; i++)
{
t.head[i-] = t.head[i];
}
t.length--;
return t;
}
int selectTable(table t,int elem)
{
for (int i=; i<t.length; i++)
{
if (t.head[i]==elem)
{
return i+;
}
}
return -;
}
table amendTable(table t, int elem, int newElem)
{
int add = selectTable(t, elem);
t.head[add-] = newElem;
return t;
}
void displayTable(table t)
{
for (int i=; i<t.length; i++)
{
printf("%d",t.head[i]);
}
printf("\n");
}
int main()
{
table t1 = initTable();
for (int i=; i<=Size; i++)
{
t1.head[i-] = i;
t1.length++;
}
printf("原顺序表:\n");
displayTable(t1);
printf("删除元素1:\n");
t1=delTable(t1, );
displayTable(t1);
printf("在第2的位置插入元素5:\n");
t1 = addTable(t1, , );
displayTable(t1);
printf("查找元素3的位置:\n");
int add = selectTable(t1, );
printf("%d\n",add);
printf("将元素3改为6:\n");
t1 = amendTable(t1, , );
displayTable(t1);
return ;
}
输出结果: 原顺序表: 删除元素1: 在第2的位置插入元素5: 查找元素3的位置: 将元素3改为6:
顺序表的优缺点
顺序表实现的基础,完全借用了数组这一数据类型,优点是在对数据进行遍历时,数据在连续的物理空间中存放,查找的速度比较快。
但是由于数组本身的限制,在向顺序表中新增或者删除数据元素时,如果被操作位置后续有很多数据元素,后续所有的数据元素都需要前移,最后虽然实现了功能,但是程序总体效率不高。
数据结构4:顺序表(线性表的顺序存储结构)及C语言实现的更多相关文章
- YTU 2987: 调整表中元素顺序(线性表)
2987: 调整表中元素顺序(线性表) 时间限制: 1 Sec 内存限制: 2 MB 提交: 1 解决: 1 题目描述 若一个线性表L采用顺序存储结构存储,其中所有元素都为整数.设计一个算法,将所 ...
- 完成代码将x插入到该顺序有序线性表中,要求该线性表依然有序
#include <stdio.h> #include <malloc.h> int main(void) { int i, n; double s = 1.3; double ...
- 数据结构线性表的动态分配顺序存储结构算法c语言具体实现和算法时间复杂度分析
#include<stdio.h>#include<stdlib.h>//线性表的动态分配顺序存储结构#define LIST_INIT_SIZE 100//线性表存储空间的初 ...
- 数据结构(Java描述)之线性表
基础概念 数据结构:是相互之间存在一种或多种关系的数据元素的集合. 逻辑结构和物理结构 关于数据结构,我们可以从逻辑结构和物理结构这两个维度去描述 逻辑结构是数据对象中数据元素之间的关系,是从逻辑意义 ...
- 数据结构与算法分析java——线性表1
说到线性结构的话,我们可以根据其实现方式分为三类: 1)顺序结构的线性表 2)链式结构的线性表 3)栈和队列的线性表 应用程序后在那个的数据大致有四种基本的逻辑结构: 集合:数据元素之间只有&qu ...
- 线性表之顺序存储结构(C语言动态数组实现)
线性表的定义:N个数据元素的有限序列 线性表从存储结构上分为:顺序存储结构(数组)和 链式存储结构(链表) 顺序存储结构:是用一段连续的内存空间存储表中的数据 L=(a1,a2,a3....an) 链 ...
- 数据结构代码整理(线性表,栈,队列,串,二叉树,图的建立和遍历stl,最小生成树prim算法)。。持续更新中。。。
//归并排序递归方法实现 #include <iostream> #include <cstdio> using namespace std; #define maxn 100 ...
- 《数据结构》C++代码 线性表
线性表,分数组和链表两种(官方名称记得是叫顺序存储和链式存储).代码里天天用,简单写写. 首先是数组,分静态.动态两种,没什么可说的,注意动态的要手动释放内存就好了. 其次是链表,依旧分静态.动态.课 ...
- [数据结构 - 第3章] 线性表之单链表(C++实现)
一.类定义 单链表类的定义如下: #ifndef SIGNALLIST_H #define SIGNALLIST_H typedef int ElemType; /* "ElemType类型 ...
随机推荐
- 侯捷STL学习(二)--序列容器测试
第六节:容器之分类和各种测试(四) stack不提供iterator操作,破坏了容器的独特性,先进先出. 使用容器multiset(允许元素重复) 内部是红黑树,insert操作就保证了排好了序. 标 ...
- 国际化---demo1---bai
login.jsp <%@ page language="java" import="java.util.*" pageEncoding="UT ...
- select *和select 全部
select *和select 全部字段 在查询上效果是一样的,速度也是一样的. 不过理论上来说select *反而会快点. 因为 1.select 全部字段在数据传输上消耗会更多,如果几百个字段这个 ...
- jQuery UI vs Kendo UI & jQuery Mobile vs Kendo UI Mobile
jQuery UI vs Kendo UI http://jqueryuivskendoui.com/#introduction jQuery Mobile vs Kendo UI Mobile ht ...
- C语言学习笔记--动态库和静态库的使用
1.C语言中的链接器 (1)每个 C 语言源文件被编译后生成目标文件,这些目标文件最终要被链接在一起生成可执行文件. (2)链接器的主要作用是把各个模块之间相互引用的部分处理好,使得各个模块之间能够正 ...
- sql 的积累
sql的积累 By:山高似水深 原创 转载注明出处 .REVERSE() 反转 例如: Hive 可用 2016年12月3日11:31:59 2.instr(str,'.')位置 结果:得出在str中 ...
- 运行程序显示:Could not find version 8.3 of the MCR.
- 无法解决 equal to 操作中 "Chinese_PRC_CI_AS" 和 "Chinese_PRC_BIN" 之间的排序规则冲
在两个数据库之间进行复合查询时有时会出现如下错误: 无法解决 equal to 操作中 "Chinese_PRC_CI_AS" 和 "Chinese_PRC_BIN&qu ...
- sql 一些偶尔会用到的写法和函数 不定时更新
小数转整数: --round() 遵循四舍五入把原值转化为指定小数位数,如: ) -- =1 ) -- =2 --floor() 向下舍入为指定小数位数 如: SELECT floor(1.45) - ...
- Python01 python入门介绍
1 python简介 1.1 为什么学python python(英国发音:/ˈpaɪθən/ 美国发音:/ˈpaɪθɑːn/), 是一种面向对象的解释型计算机程序设计语言,由荷兰人Guido van ...