【数据结构】--顺序表的实现

这篇具有很好参考价值的文章主要介绍了【数据结构】--顺序表的实现。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、顺序表的概念

什么是顺序表?顺序表(SeqList)是线性表中的一类。而线性表是n个具有相同特性的数据元素的有限序列。线性表是一种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、字符串、栈、队列... 注意:线性表在逻辑上是线性结构,也就是说是一条连续的直线。但在物理结构上不一定是连续的。(这里很好理解,例如有10个人在买蜜雪冰城,在逻辑上应该是10个人按照先后顺序排成一队一个一个买,突然这10个人中的一个人的朋友也要买蜜雪冰城,但是他不想排队,他就站在他朋友的旁边一起买。这就是在物理上不一定连续。)

而顺序表是用一组地址连续的存储单元依次存储线性表中的数据元素。顺序表在逻辑结构是线性的结构,在物理结构上是连续的。它的底层是基于数组来实现的,所以顺序表的实现其实是在数组上完成数据的增删查改。

二、顺序表的分类

1、静态顺序表   

 定义:使用定长数组存储数据 

//静态顺序表
typedef int SLDataType;
#define N 6
typedef struct SeqList
{
    SLDataTyppe arr[N];//定长数组
    int size;//有效数据个数
}SL;

2、动态顺序表(推荐使用)

 定义:按需申请容量大小

//动态顺序表

//顺序表不仅可以插入整形数据,还可以插入浮点型,char型等等
//所以我们给int重新定义一个名字SLDataType,后面如果想要换数据类型,直接更改typedef后面的int就好了
typedef int SLDataType;

typedef struct SeqList
{
    SLDataType* arr;//一级指针指向的动态开辟的数组
    int size;//有效数据个数
    int capacity;//数组的容量大小
}SL;

在这里我推荐使用动态顺序表。静态顺序表如果空间给少了不够用,给多了会造成空间浪费。动态顺序表可以根据需要的多少来调整顺序表的大小。 

三、动态顺序表的实现

1、定义一个动态顺序表   

//动态顺序表
typedef int SLDataType;

typedef struct SeqList
{
    SLDataType* arr;//一级指针指向的动态开辟的数组
    int size;//有效数据个数
    int capacity;//数组的容量大小
}SL;

2、对动态顺序表进行初始化

//初始化顺序表
void SeqListInit(SL* ps)
{
	assert(ps);//断言ps指针不能是空指针
	ps->arr = NULL;
	ps->capacity = ps->size = 0;
}

3、对顺序表进行扩容 

                                扩容思路:

(1)判断顺序表的空间容量是否足够?当顺序表的数据个数==空间容量时表示顺序表的空间容量已经满了,需要进行扩容。

(2)用三目表达式来判断,如果顺序表的空间容量为0,则先给空间容量为4(这个数值是随便设置);如果空间容量不为0,则让空间容量以2倍的数量进行增加。(这样可以在扩容的同时减少空间的浪费)

(3)用realloc进行动态空间的扩容。

(4)检查realloc是否扩容成功并用一个临时指针temp来接收,若temp不为空,则将temp赋值给ps->arr,并将新的空间容量newcapacity赋值给capacity。

//扩容
void SLCheckCapacity(SL* ps)
{
	assert(ps);
	if (ps->capacity == ps->size)
	{
		//扩容
		int newcapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity;
		SLDataType* temp = (SLDataType*)realloc(ps->arr, newcapacity * sizeof(SLDataType));
		if (temp == NULL)
		{
			perror("realloc fail!");
			exit(1);//直接退出程序不在执行
		}
		ps->arr = temp;
		ps->capacity = newcapacity;
	}
}

4、打印顺序表

顺序表的底层是数组,所以打印顺序表就相当于打印数组,这里非常简单所以直接上代码

//打印顺序表
void SeqListPrint(SL* ps)
{
	assert(ps);
	int i = 0;
	for (i = 0; i < ps->size; i++)
	{
		printf("%d ", ps->arr[i]);
	}
	printf("\n");
}

5、顺序表尾插数据

顺序表尾插就是在数组里size这个位置插入数据,所以需要注意的当插入完数据之后不要忘了size++。

【数据结构】--顺序表的实现,数据结构,数据结构

例如这里尾插一个4,在ps->arr[ps->size]这个位置插入完数据之后,ps->size要往后移动一位。

//尾插
void SeqListPushBack(SL* ps, SLDataType x)
{
	assert(ps);
	//判断是否需要扩容
	SLCheckCapacity(ps);
	ps->arr[ps->size] = x;
	ps->size++;
}

6、顺序表头插数据

顺序表头插其实就是在数组的下标为0的位置插入数据,所以我们需要先把顺序表里面的数据从后往前依次向后移动一位,把下标为0的位置空出来在插入数据。

【数据结构】--顺序表的实现,数据结构,数据结构

例如我想头插一个5,所以我需要先把1,2,3,4四个数据从后往前依次往后移动一位,然后插入5。

//头插
void SeqListPushFront(SL* ps,SLDataType x)
{
	assert(ps);
	//判断是否需要扩容
	SLCheckCapacity(ps);
	int i = 0;
	for (i = ps->size; i > 0; i--)
	{
		ps->arr[i] = ps->arr[i - 1];
	}
	ps->arr[0] = x;
	ps->size++;
}

7、顺序表尾删数据

顺序表尾删数据就非常简单了,只需要让ps->size--,size向前移动一位就好了。但需要注意的一点是必须要判断一下ps->size是否大于0,因为顺序表里必须要有数据才能够做删除操作。这里直接上代码。

//尾删
void SeqListPopBack(SL* ps)
{
	assert(ps);
	//这里有两种方法来判断ps->size是否>0
	//第一种方法比较温柔
	//if (ps->size == 0)
	//{
		//return;
	//}
	//第二种方法比较简单粗暴
	assert(ps->size > 0);
	ps->size--;
}

8、顺序表头删数据

头删操作也非常的easy,头删只需要把顺序表里从下标为1的数据开始依次往前移动一位,同时也要判断ps->size是否大于0,以及删除一个数据要让ps->size--。

//头删
void SeqListPopFront(SL* ps)
{
	assert(ps);
	assert(ps->size > 0);
	int i = 0;
	for (i = 1; i < ps->size; i++)
	{
		ps->arr[i - 1] = ps->arr[i];
	}
	ps->size--;
}

9、顺序表查找数据

顺序表查找就是通过遍历数组来比较数组里的数据与已知要查找的数据是否相等。若相等,则返回数组下标,若不相等,则返回-1表示所查找的数据不存在。

//查找顺序表
int SeqListFind(SL* ps, SLDataType x)
{
	assert(ps);
	int i = 0;
	for (i = 0; i < ps->size; i++)
	{
		if (ps->arr[i] == x)
		{
			return i;
		}
	}
	return -1;
}

10、在指定位置(pos)插入数据

在指定位置插入数据,首先我们需要通过SeqListFind函数接口来得到需要插入数据的位置下标,接着我们要把从pos到ps->size-1位置上的数据从后往前向后移动一位,把pos 位置空出来,最后在pos位置插入数据。

【数据结构】--顺序表的实现,数据结构,数据结构

【数据结构】--顺序表的实现,数据结构,数据结构

例如,我想要在2这个位置插入100,首先把2,3,4三个数据从后往前依次向后移动一位,然后在pos位置插入100。

//指定位置插入数据
void SeqListInsert(SL* ps, int pos, SLDataType x)
{
	assert(ps);
	//pos必须是有效的数据下标
	assert(pos >= 0 && pos < ps->size);
	//判断是否需要扩容
	SLCheckCapacity(ps);
	int i = 0;
	for (i = ps->size; i > pos; i--)
	{
		ps->arr[i] = ps->arr[i - 1];
	}
	ps->arr[pos] = x;
	ps->size++;
}

11、在指定位置(pos)删除数据

这里就很简单了,只需要让pos位置之后的数据整体向前移动一位就好了。

//指定位置删除数据
void SeqListErase(SL* ps, int pos)
{
	assert(ps);
	//pos必须是有效的数据下标
	assert(pos >= 0 && pos < ps->size);
	int i = 0;
	for (i = pos; i < ps->size; i++)
	{
		ps->arr[i] = ps->arr[i + 1];
	}
	ps->size--;
}

12、销毁顺序表

//销毁顺序表
void SeqListDestroy(SL* ps)
{
	assert(ps);
	free(ps->arr);
	ps->arr = NULL;
	ps->capacity = ps->size = 0;
}

四、完整代码

SeqList.h

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>

typedef int SLDataType;

typedef struct SeqList
{
	SLDataType* arr;
	int size;
	int capacity;
}SL;

//初始化顺序表
void SeqListInit(SL* ps);
//销毁顺序表
void SeqListDestroy(SL* ps);
//扩容
void SLCheckCapacity(SL* ps);
//打印顺序表
void SeqListPrint(SL* ps);
//尾插
void SeqListPushBack(SL* ps, SLDataType x);
//头插
void SeqListPushFront(SL* ps);
//尾删
void SeqListPopBack(SL* ps);
//头删
void SeqListPopFront(SL* ps);
//查找顺序表
int SeqListFind(SL* ps, SLDataType x);
//指定位置之前插入数据
void SeqListInsert(SL* ps, int pos, SLDataType x);
//指定位置删除数据
void SeqListErase(SL* ps, int pos);

SeqList.c

#define _CRT_SECURE_NO_WARNINGS 1
#include"SeqList.h"

//初始化顺序表
void SeqListInit(SL* ps)
{
	assert(ps);
	ps->arr = NULL;
	ps->capacity = ps->size = 0;
}

//销毁顺序表
void SeqListDestroy(SL* ps)
{
	assert(ps);
	free(ps->arr);
	ps->arr = NULL;
	ps->capacity = ps->size = 0;
}

//扩容
void SLCheckCapacity(SL* ps)
{
	assert(ps);
	if (ps->capacity == ps->size)
	{
		//扩容
		int newcapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity;
		SLDataType* temp = (SLDataType*)realloc(ps->arr, newcapacity * sizeof(SLDataType));
		if (temp == NULL)
		{
			perror("realloc fail!");
			exit(1);//直接退出程序不在执行
		}
		ps->arr = temp;
		ps->capacity = newcapacity;
	}
}

//打印顺序表
void SeqListPrint(SL* ps)
{
	assert(ps);
	int i = 0;
	for (i = 0; i < ps->size; i++)
	{
		printf("%d ", ps->arr[i]);
	}
	printf("\n");
}

//尾插
void SeqListPushBack(SL* ps, SLDataType x)
{
	assert(ps);
	//判断是否需要扩容
	SLCheckCapacity(ps);
	ps->arr[ps->size] = x;
	ps->size++;
}

//头插
void SeqListPushFront(SL* ps,SLDataType x)
{
	assert(ps);
	//判断是否需要扩容
	SLCheckCapacity(ps);
	int i = 0;
	for (i = ps->size; i > 0; i--)
	{
		ps->arr[i] = ps->arr[i - 1];
	}
	ps->arr[0] = x;
	ps->size++;
}

//尾删
void SeqListPopBack(SL* ps)
{
	assert(ps);
	//这里有两种方法来判断ps->size是否>0
	//第一种方法比较温柔
	//if (ps->size == 0)
	//{
		//return;
	//}
	//第二种方法比较简单粗暴
	assert(ps->size > 0);
	ps->size--;
}

//头删
void SeqListPopFront(SL* ps)
{
	assert(ps);
	assert(ps->size > 0);
	int i = 0;
	for (i = 1; i < ps->size; i++)
	{
		ps->arr[i - 1] = ps->arr[i];
	}
	ps->size--;
}

//查找顺序表
int SeqListFind(SL* ps, SLDataType x)
{
	assert(ps);
	int i = 0;
	for (i = 0; i < ps->size; i++)
	{
		if (ps->arr[i] == x)
		{
			return i;
		}
	}
	return -1;
}

//指定位置插入数据
void SeqListInsert(SL* ps, int pos, SLDataType x)
{
	assert(ps);
	//pos必须是有效的数据下标
	assert(pos >= 0 && pos < ps->size);
	//判断是否需要扩容
	SLCheckCapacity(ps);
	int i = 0;
	for (i = ps->size; i > pos; i--)
	{
		ps->arr[i] = ps->arr[i - 1];
	}
	ps->arr[pos] = x;
	ps->size++;
}

//指定位置删除数据
void SeqListErase(SL* ps, int pos)
{
	assert(ps);
	//pos必须是有效的数据下标
	assert(pos >= 0 && pos < ps->size);
	int i = 0;
	for (i = pos; i < ps->size; i++)
	{
		ps->arr[i] = ps->arr[i + 1];
	}
	ps->size--;
}

test.c

测试文件可以自己写,这里仅供参考。

#define _CRT_SECURE_NO_WARNINGS 1
#include"SeqList.h"

void test()
{
	SL s;
	SeqListInit(&s);
	SeqListPushBack(&s, 1);
	SeqListPushBack(&s, 2);
	SeqListPushBack(&s, 3);
	SeqListPushBack(&s, 4);
	SeqListPrint(&s);
	//SeqListPushFront(&s, 5);
	//SeqListPrint(&s);
	//SeqListPopBack(&s);
	//SeqListPrint(&s);
	//SeqListPopFront(&s);
	//SeqListPopFront(&s);
	//SeqListPopFront(&s);
	//SeqListPopFront(&s);
	//SeqListPopFront(&s);
	//SeqListPrint(&s);
	int find = SeqListFind(&s, 2);
	if (find < 0)
	{
		printf("所查找的数据不存在!\n");
	}
	else
	{
		printf("所查找的数据存在,该数据所在位置下标为%d\n", find);
	}
	SeqListInsert(&s, find, 100);
	SeqListPrint(&s);
	SeqListErase(&s, find);
	SeqListPrint(&s);
}

int main()
{
	test();
	return 0;
}

这是关于数据结构的第一篇博客,后期我会持续更新数据结构以及C语言的相关内容。欢迎大家在评论区相互讨论以及指出我的不足,我会在后面的文章中参考大家的建议来写出更好的博客。

感谢大家点赞+关注,来跟着我一起学习吧!

 文章来源地址https://www.toymoban.com/news/detail-854972.html

 

到了这里,关于【数据结构】--顺序表的实现的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

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

    线性表的顺序存储结构称为 顺序表 ,其基本思想是 用一段地址连续的存储单元一次存储线性表的数据元素。 设顺序表的每个元素占用 c 个存储单元,则第 i 个元素的存储地址为: 所以, 只要确定了存储顺序表的起始地址(即基地址),计算任意一个元素的存储地址的时间

    2024年03月15日
    浏览(53)
  • 【(数据结构) —— 顺序表的应用-通讯录的实现】

    C语言基础要求:结构体、动态内存管理、顺序表、文件件操作 (1). 功能要求 1)至少能够存储100个人的通讯信息 2)能够保存用户信息:名字、性别、年龄、电话、地址等 3)增加联系人信息 4)删除指定联系人 5)查找制定联系人 6)修改指定联系人 7)显示联系人信息 (2).重

    2024年02月08日
    浏览(48)
  • 数据结构之顺序表的实现(详解!附完整代码)

    线性表(linear list)是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使用的数据结构 常见的线性表:顺序表、链表、栈、队列、字符串… 线性表在逻辑上是线性结构,也就说是连续的一条直线。但是在物理结构上并不一定是连续的,线性表在物理上存

    2024年02月04日
    浏览(40)
  • 数据结构之顺序表的实现(C语言版)

         Hello, 大家好,我是一代,今天给大家带来有关顺序表的有关知识      所属专栏:数据结构      创作不易,望得到各位佬们的互三呦 1.首先在讲顺序表之前我们先来了解什么是数据结构 数据结构是由“数据”和“结构”两词组合⽽来。 什么是数据?常见的数值1、

    2024年04月25日
    浏览(47)
  • C语言数据结构-----顺序表(多功能动态顺序表的代码实现)

    本篇讲述了顺序表的相关知识,以及动态顺序表的代码实现。 顺序表和链表一般情况下都会叫他们线性表。 线性表(linear list)是n个具有相同特性的数据元素的有限序列。线性表是一种在实际中广泛使 用的数据结构,常见的线性表:顺序表、链表、栈、队列、字符串… 线性

    2024年02月07日
    浏览(48)
  • [C语言][数据结构][动态内存空间的开辟]顺序表的实现!

    目录 零.必备知识 a.顺序表的底层是数组. b.数组在内存中是连续存放的. c.动态内存空间的开辟(malloc,calloc,realloc). 一.顺序表的定义与实现          1.1 顺序表的定义          1.2 顺序表的初始化          1.3 顺序表的销毁          1.4 顺序表容量的检查与调整

    2024年04月09日
    浏览(88)
  • 数据结构(C语言实现)——顺序表的介绍及基本操作的实现

    今天我们来学习数据结构中的线性表,本文主要介绍一种常见的线性表——顺序表。 本文着重介绍顺序表的概念以及顺序表各种基本操作的实现过程(C语言实现),以后会更新更多的数据结构,觉得有用的朋友可以三连关注一波,一起学习。 线性表(linear list)是n个具有相

    2023年04月13日
    浏览(52)
  • 【数据结构】顺序表的实现及基本操作完整代码(C语言实现)

    顺序表:逻辑上相邻的数据元素,其物理次序也是相邻的 这里之所以要把int分别创建新名字为SqlElemType和Status,是因为实际应用时数据的类型不一定是int型,这样设置方便修改元素类型,提高代码适用性。 LocateElem的时间复杂度为O(n) InsertSq的时间复杂度为O(n) DeleteSq的时间

    2024年04月12日
    浏览(50)
  • 数据结构-顺序表的基本实现(C语言,简单易懂,含全部代码)

    今天起开始编写数据结构中的各种数据结构及算法的实现,说到顺序表,我们首先得了解下线性表。 线性表(linear list)是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列、字符串… 线性表在逻

    2023年04月08日
    浏览(41)
  • 数据结构-线性表的顺序表基本操作代码实现(超级详细清晰 C++实现)

    顺序表是用一段 物理地址连续的存储单元 依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成数据的增删查改。 顺序表: 可动态增长的数组,要求数据是连续存储的 特点: 随机访问 顺序既可以 静态分配 ,也可以 动态分配 。在静态分配时,由于数组

    2024年02月07日
    浏览(56)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包