草庐IT

【数据结构】线性表的顺序存储结构及实现——C语言版

刹那芳间- 2023-09-11 原文



文章目录


顺序表

线性表的顺序存储结构称为顺序表,其基本思想是用一段地址连续的存储单元一次存储线性表的数据元素。

设顺序表的每个元素占用 c 个存储单元,则第 i 个元素的存储地址为:

所以,只要确定了存储顺序表的起始地址(即基地址),计算任意一个元素的存储地址的时间是相等的。

我们通常用一维数组来实现顺序表,也就是把线性表中相邻的元素存储在数组中相邻的位置,从而导致数据元素的序号和存放它的数组下标之间具有一一对应的关系。

注意: \color{RoyalBlue}注意: 注意:
C语言中数组的下标是 从0开始 的,而顺序表中元素的序号是 从1开始 的,即线性表中第 i 个元素存储在数组中下标为 i-1 的位置。

定义一个数组必须确定数组的长度。由于线性表中可以进行插入操作,所以数组长度要大于当前线性表的长度。MaxSize表示数组长度,用length表示线性表的长度。

1. 顺序表的存储结构定义

#define MaxSize 100  //假设顺序表最多存放100个元素
typedef int DataType;	//定义线性表的数据类型,假设为int型

typedef struct
{
	DataType data[MaxSize];	//存放数据元素的数组
	int length;	//线性表的长度
}SeqList;

2. 顺序表的实现

2.1 初始化顺序表

初始化顺序表只需要将顺序表的长度length初始化为0,

void InitList(SeqList* L)
{
	L->length = 0;
}

2.2 建立顺序表

建立一个长度为n的顺序表,需要将给定的数据元素传入顺序表中,并将传入的元素个数作为顺序表的长度。

设给定的数据元素存放在数组a[n]中,建立顺序表的操作如图:

函数CreatList的返回值表示建立顺序表操作是否成功,如果顺序表的存储空间小于给定数据元素个数,则无法建立顺序表。

int CreatList(SeqList* L, DataType a[], int n)
{
	if (n > MaxSize)
	{
		printf("顺序表的存储空间不够,无法建立顺序表\n");
		return 0;
	}
	for (int i = 0; i < n; i++)
	{
		L->data[i] = a[i];
	}
	L->length = n;
	return 1;
}

2.3 销毁顺序表

顺序表是静态存储分配,在顺序表变量退出作用域时,自动释放该变量所占内存单元。因此,顺序表无须销毁。

2.4 判空操作

顺序表的判空操作只需要判断长度length是否为0就可以了,

int Empty(SeqList* L)
{
	if (L->length == 0) {
		return 1;	//顺序表为空返回1
	}
	else
	{
		return 0;
	}
}

2.5 求顺序表的长度

int Length(SeqList* L)
{
	return L->length;
}

2.6 遍历操作

在顺序表中,遍历操作即是按下标依次输出各元素

void PrintList(SeqList* L)
{
	for (int i = 0; i < L->length; i++)
	{
		printf("%d ", L->data[i]);	//输出线性表的元素值,假设为int型
	}
}

2.7 按值查找

在顺序表中实现按值查找操作,需要对顺序表中的元素依次进行比较,如果查找成功,返回元素的序号(注意不是下标),否则返回0

int Locate(SeqList* L, DataType x)
{
	for (int i = 0; i < L->length; i++)
	{
		if (L->data[i] == x)
		{
			return i + 1;	//返回序号
		}
	}
	return 0;	//循环结束,说明查找失败
}

2.8 按位查找

顺序表中第i个元素存储在数组中下标为i-1的位置,
所以,很容易实现按位查找。

函数Get的返回值表示是否查找成功,若查找成功,通过指针参数ptr返回查找到的元素值。

int Get(SeqList* L, int i, DataType* ptr)
{
	if (i<1 || i>L->length)
	{
		printf("查找位置非法,查找失败\n");
		return 0;
	}
	else
	{
		*ptr = L->data[i - 1];
		return 1;
	}
}

2.9 插入操作

插入操作是在表的第i(1≦i≦n+1)个位置进行插入新元素x,使长度为n的线性表变成了长度为n+1的线性表。

注意: \color{RoyalBlue}注意: 注意:
元素移动的方向,是从最后一个元素开始移动,直至将第i个元素后移为止,然后将新元素插入i处。如果表满,则引发上溢错误,如果元素的插入位置不合法,则引发位置错误。

int Insert(SeqList* L, int i, DataType x)
{
	if (L->length >= MaxSize)
	{
		printf("上溢错误,插入失败");
		return 0;
	}
	if (i<1 || i>L->length + 1)
	{
		printf("位置错误,插入失败");
		return 0;
	}
	for (int j = L->length; j >= i; j--)
	{
		L->data[j] = L->data[j - 1];
	}
	L->data[i - 1] = x;
	L->length++;
	return 1;
}

2.10 删除操作

删除操作是将表的第i(1≦i≦n)个元素删除,使长度为n的线性表变成了长度为n-1的线性表。

注意: \color{RoyalBlue}注意: 注意:
元素移动的方向,是从第 i+1 个元素(下标为i)开始移动,直至将最后一个元素前移为止,并且在移动元素之前取出被删元素。如果表空,则引发下溢错误,如果元素的删除位置不合理,则引发位置错误。

int Delete(SeqList* L, int i, DataType* ptr)
{
	if (L->length == 0)
	{
		printf("下溢错误,删除失败\n");
		return 0;
	}
	if (i<1 || i>L->length)
	{
		printf("位置错误,删除失败\n");
		return 0;
	}
	*ptr = L->data[i - 1];	//取出位置i的元素
	for (int j = i ; j < L->length; j++)
	{
		L->data[j - 1] = L->data[j];
	}
	L->length--;
	return 1;
}

3. 顺序表的使用

#include<stdio.h>
#include<stdlib.h>
/*将顺序表的存储结构定义和各个函数定义放到这里*/

int main()
{
	int r[5] = { 1,2,3,4,5 }, i, x;
	SeqList L;	//定义变量L为顺序表类型
	CreatList(&L, r, 5);	//建立具有5个元素的顺序表
	printf("当前线性表的数据为:");
	PrintList(&L);	//输出当前线性表 1 2 3 4 5
	Insert(&L, 2, 8);	//在第2个位置插入值为8的元素
	printf("插入之后的数据为:");
	PrintList(&L);	//输出插入后的线性表 1 8 2 3 4 5
	printf("当前线性表的长度为:%d\n", Length(&L));	//输出线性表的长度6
	printf("请输入查找的元素值:");
	scanf("%d", &x);
	i = Locate(&L, x);
	if (i == 0)
	{
		printf("查找失败\n");
	}
	else
	{
		printf("元素%d的位置为:%d\n", x, i);
	}
	printf("请输入查找第几个元素的值:");
	scanf("%d", &i);
	if (Get(&L, i, &x) == 1)
	{
		printf("第%d个元素的值为:%d\n", i, x);
	}
	else
	{
		printf("线性表中没有第%d个位置元素\n",i);
	}
	printf("请输入要删除第几个元素:");
	scanf("%d", &i);
	if (Delete(&L, i, &x) == 1)
	{
		printf("删除成功,删除的数据为%d\n", x);
	}
	else
	{
		printf("删除操作失败\n");
	}
	return 0;
}

4. 暖暖树洞

“要留点精力去读书去运动去爱人,去奔赴想要的生活,不应该把精力浪费在痛苦的社交讨厌的人那里,看起来可以挽回的事情,仔细想想一点都不值得,贪恋过去的快乐注定走不远,过去的就让它过去吧,在热爱生活的同时快乐的小事情真的很多很多。”

有关【数据结构】线性表的顺序存储结构及实现——C语言版的更多相关文章

  1. ruby - 使用 ruby​​ 将 HTML 转换为纯文本并维护结构/格式 - 2

    我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h

  2. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  3. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  4. ruby - Chef 执行非顺序配方 - 2

    我遵循了教程http://gettingstartedwithchef.com/,第1章。我的运行list是"run_list":["recipe[apt]","recipe[phpap]"]我的phpapRecipe默认Recipeinclude_recipe"apache2"include_recipe"build-essential"include_recipe"openssl"include_recipe"mysql::client"include_recipe"mysql::server"include_recipe"php"include_recipe"php::modul

  5. ruby - 如何根据特征实现 FactoryGirl 的条件行为 - 2

    我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden

  6. ruby - Ruby 有 `Pair` 数据类型吗? - 2

    有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳

  7. ruby - 寻找通过阅读代码确定编程语言的ruby gem? - 2

    几个月前,我读了一篇关于ruby​​gem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:

  8. ruby - 是否有用于序列化和反序列化各种格式的对象层次结构的模式? - 2

    给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最

  9. ruby - 我如何添加二进制数据来遏制 POST - 2

    我正在尝试使用Curbgem执行以下POST以解析云curl-XPOST\-H"X-Parse-Application-Id:PARSE_APP_ID"\-H"X-Parse-REST-API-Key:PARSE_API_KEY"\-H"Content-Type:image/jpeg"\--data-binary'@myPicture.jpg'\https://api.parse.com/1/files/pic.jpg用这个:curl=Curl::Easy.new("https://api.parse.com/1/files/lion.jpg")curl.multipart_form_

  10. 世界前沿3D开发引擎HOOPS全面讲解——集3D数据读取、3D图形渲染、3D数据发布于一体的全新3D应用开发工具 - 2

    无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD

随机推荐