数据结构初阶(C语言)-顺序表

一,线性表

在进行顺序表的介绍之前,我们先来了解下什么是线性表:

线性表是n个具有相同特性的数据元素的有限序列。 线性表是⼀种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列、字符串...线性表在逻辑上是线性结构,也就说是连续的⼀条直线。但是在物理结构上并不⼀定是连续的, 线性表在物理上存储时,通常以数组和链式结构的形式存储。

二,顺序表

2.1概念与其逻辑及物理结构

概念:顺序表是用⼀段物理地址连续的存储单元依次存储数据元素的线性结构,⼀般情况下采用数组存储。

这里我们可以得知顺序表的逻辑结构(人为想象的结构)呈线性,同时它的物理结构是由连续的数组构成,所以它的物理结构也为线性。

2.2静态顺序表

概念:使用已经确定长度的数组存储元素,这里我们用代码来表示静态顺序表的结构:

cs 复制代码
#define N 7
typedef int SLDataType;
typedef struct SeqList
{
	SLDataType arr[N];
	int size;//有效的数据个数
};

数组的长度我们无法改变,也就意味着我们无法随意的增加和删除数据,更不能在数据量变大时对储存数据的顺序表进行扩容,所以我们主要使用的顺序表为动态顺序表。

2.3动态顺序表

概念:使用可变长度的数组存储元素,这里我们使用代码来表示动态顺序表的基本结构:

cpp 复制代码
typedef int SLDataType;

typedef struct SepList
{
	SLDataType* arr;
	int capacity;
	int size;
}SL,*PSL;

三,动态顺序表的功能实现

顺序表的目的是为了存储数据,所以我们这里所说的实现主要有头,尾插法。头,尾删法。定位插,删法。不过在此之前,如果空间的大小不够,我们将无法对数据进行增加,所以在进行所有的实现之前,我们需要先对空间大小进行检查:

3.1检查内存空间大小是否足够

我们在扩充内存空间时,经常以原空间大小的倍数来扩充,这是因为随着我们扩充的次数增多,我们所需要存储的数据量也会按倍数增大,所以当我们以倍数来扩充内存空间时,正好与数据的增长量成线性关系,浪费的空间也将最大化的被缩小。通常情况下,储存空间我们以上次纯村空间的大小的二倍来增加,当然在此之前我们要对顺序表进行初始化:

cs 复制代码
void SepInit(PSL ps)
{
	ps->arr = NULL;
	ps->capacity = ps->size = 0;
}

对空间的大小检查的代码如下:

cs 复制代码
void CheckSpace(PSL ps)
{
	if (ps->capacity == ps->size)
	{
		int excapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
		SLDataType* tmp = (SLDataType*)realloc(ps->arr, excapacity * sizeof(SLDataType));
		if (tmp == NULL)
		{
			perror("SLInsertEnd,tmp:");
			return 1;
		}
		ps->arr = tmp;
		ps->capacity = excapacity;
	}
}

这里我们需要对源顺序表进行检验,若它刚经过初始化,我们为它付一个初始值为4的容量(为什么不在初始化中对其进行赋值,我个人认为是因为后续对数据的增删可能会使顺序表变为空,在这里对其容量进行改动是为了避免这种情况),剩下就是正常的开辟顺序表的空间(使用realloc,因为realloc在对空指针增加空间后会返回一个新的地址对应开辟的空间,后续我们也需要对内存进行扩充,所以使用realloc最合适)。

3.2尾删法

顾名思义就是从尾部减少有效数据的个数,实现代码非常简单:

cs 复制代码
void SLDeltEnd(PSL ps)
{
	assert(ps);
	ps->size--;
}

3.3头删法

这里我们需要从头部删除数据,其实就是把除了第一个数据外的所有数据整体向前移动一格,进而覆盖第一个数据,达到头删的目的,实现代码如下:

cs 复制代码
void SLDeltBeg(PSL ps)
{
	assert(ps);
	assert(ps->size);
	for (int i = 0; i < ps->size - 1; i++)
	{
		ps->arr[i] = ps->arr[i + 1];
	}
	ps->size--;
}

当然我们需要判断有效数据的个数,如果为0就没有删除的必要了。

3.4尾插法

从尾部插入,其实就是向数组中的第size个位置插入数据,但如果数据在此之前已满,我们则需要扩充空间,具体实现代码如下:

cs 复制代码
void SLInsertEnd(PSL ps,SLDataType x)
{

	assert(ps);
	CheckSpace(ps);
	ps->arr[ps->size++] = x;
}

3.5头插法

有了上面头删法的帮助,其实我们这里也就不难理解头删法是如何实现的,其实就是将现有数据整体向后移动一个位置,然后在首元素位置插入我们的数据,实现代码如下:

cs 复制代码
void SLInsertBeg(PSL ps,SLDataType x)
{
	assert(ps);
	CheckSpace(ps);
	for (int i = ps->size; i > 0; i--)
	{
		ps->arr[i] = ps->arr[i - 1];
	}
	ps->arr[0] = x;
	ps->size++;
}

3.6定位删除法

在这里我们需要注意,定位的方式删除与增加其实基本上沿用了头删与头插的思想,无非就是在传参时我们需要多传一个参数--需要删除或增加的位置,删除则将除了当前位置之后的所有元素整体向前移动一个位置,而增加则是将当前指定位置数据及之后的数据整体向后移动一个位置,然后再在当前位置放上我们的目标数据:

cs 复制代码
void SLDeltDes(PSL ps, int pos)
{
	assert(ps);
	assert(pos>=0 && pos<ps->size);
	for (int i = pos; i < ps->size - 1; i++)
	{
		ps->arr[i] = ps->arr[i + 1];
	}
	ps->size--;
}

3.7定位插入法

有了上面3.6的介绍,我们这里也不废话,直接上实现代码:

cs 复制代码
void SLDeltDes(PSL ps, int pos)
{
	assert(ps);
	assert(pos>=0 && pos<ps->size);
	for (int i = pos; i < ps->size - 1; i++)
	{
		ps->arr[i] = ps->arr[i + 1];
	}
	ps->size--;
}

当然删除肯定要有数据才能删,有那个位置的空间才能定位增加数据,这也是我们上面使用assert断言的原因。

四,实现功能后的善后处理

4.1打印顺序表内容

cs 复制代码
void SLPrintf(PSL ps)
{
	for (int i = 0; i < ps->size; i++)
	{
		printf("%d ",ps->arr[i]);
	}
	printf("\n");
}

4.2释放顺序表

cs 复制代码
void SLDestory(PSL ps)
{
	if (ps->arr)
	{
		free(ps->arr);
	}
	ps->capacity = 0;
	ps->size = 0;
	ps->arr = NULL;
}

4.3动态顺序表功能实现的完整代码

cs 复制代码
//头文件SepList.h
#pragma once
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>

typedef int SLDataType;

typedef struct SepList
{
	SLDataType* arr;
	int capacity;
	int size;
}SL,*PSL;

void SepInit(PSL ps);//顺序表空间的初始化

void SLInsertEnd(PSL ps,SLDataType x);//尾插法

void SLPrintf(PSL ps);//检验打印

void SLDestory(PSL ps);//释放开辟的动态内存

void CheckSpace(PSL ps);//检查空间是否足够

void SLInsertBeg(PSL ps,SLDataType x);//头插法

void SLDeltBeg(PSL ps);//头删法

void SLDeltEnd(PSL ps);//尾删法

void SLDeltDes(PSL ps, int pos);//指定位置删除

void SLInsertDes(PSL ps, int pos, SLDataType x);//指定位置增加数据

//顺序表功能实现代码SepList.c
#include "Seplist.h"

void SepInit(PSL ps)
{
	ps->arr = NULL;
	ps->capacity = ps->size = 0;
}

void CheckSpace(PSL ps)
{
	if (ps->capacity == ps->size)
	{
		int excapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
		SLDataType* tmp = (SLDataType*)realloc(ps->arr, excapacity * sizeof(SLDataType));
		if (tmp == NULL)
		{
			perror("SLInsertEnd,tmp:");
			return 1;
		}
		ps->arr = tmp;
		ps->capacity = excapacity;
	}
}

void SLInsertEnd(PSL ps,SLDataType x)
{

	assert(ps);
	CheckSpace(ps);
	ps->arr[ps->size++] = x;
}

void SLInsertBeg(PSL ps,SLDataType x)
{
	assert(ps);
	CheckSpace(ps);
	for (int i = ps->size; i > 0; i--)
	{
		ps->arr[i] = ps->arr[i - 1];
	}
	ps->arr[0] = x;
	ps->size++;
}

void SLPrintf(PSL ps)
{
	for (int i = 0; i < ps->size; i++)
	{
		printf("%d ",ps->arr[i]);
	}
	printf("\n");
}

void SLDestory(PSL ps)
{
	if (ps->arr)
	{
		free(ps->arr);
	}
	ps->capacity = 0;
	ps->size = 0;
	ps->arr = NULL;
}

void SLDeltBeg(PSL ps)
{
	assert(ps);
	assert(ps->size);
	for (int i = 0; i < ps->size - 1; i++)
	{
		ps->arr[i] = ps->arr[i + 1];
	}
	ps->size--;
}

void SLDeltEnd(PSL ps)
{
	assert(ps);
	ps->size--;
}

void SLDeltDes(PSL ps, int pos)
{
	assert(ps);
	assert(pos>=0 && pos<ps->size);
	for (int i = pos; i < ps->size - 1; i++)
	{
		ps->arr[i] = ps->arr[i + 1];
	}
	ps->size--;
}

void SLInsertDes(PSL ps, int pos, SLDataType x)
{
	assert(ps);
	assert(pos >= 0 && pos <= ps->size);
	CheckSpace(ps);
	for (int i = ps->size; i > pos; i--)
	{
		ps->arr[i] = ps->arr[i - 1];
	}
	ps->arr[pos] = x;
	ps->size++;
}


//顺序表功能测试代码

#include "Seplist.h"

//顺序表的头插尾插以及头删尾删

void test()
{
	SL s;
	SepInit(&s);
	SLInsertEnd(&s,4);
	SLInsertEnd(&s,5);
	SLInsertEnd(&s,3);
	SLInsertEnd(&s,2);
	SLInsertEnd(&s,1);
	SLInsertBeg(&s, 1);
	SLInsertBeg(&s, 2);
	SLInsertBeg(&s, 3);
	SLInsertBeg(&s, 4);
	SLDeltBeg(&s);
	SLDeltEnd(&s);
	SLDeltDes(&s, 2);
	SLInsertDes(&s, 4, 2);
	SLPrintf(&s);
	SLDestory(&s);
}


int main()
{
	test();
	return 0;
}
相关推荐
世伟爱吗喽30 分钟前
NUXT3学习日记四(路由中间件、导航守卫)
学习
Heisenberg~34 分钟前
详解八大排序(五)------(计数排序,时间复杂度)
c语言·数据结构·排序算法
Elihuss1 小时前
ONVIF协议操作摄像头方法
开发语言·php
飞凌嵌入式2 小时前
飞凌嵌入式旗下教育品牌ElfBoard与西安科技大学共建「科教融合基地」
嵌入式硬件·学习·嵌入式·飞凌嵌入式
lb36363636364 小时前
分享一下arr的意义(c基础)(必看)(牢记)
c语言·知识点
Swift社区5 小时前
在 Swift 中实现字符串分割问题:以字典中的单词构造句子
开发语言·ios·swift
没头脑的ht5 小时前
Swift内存访问冲突
开发语言·ios·swift
没头脑的ht5 小时前
Swift闭包的本质
开发语言·ios·swift
wjs20245 小时前
Swift 数组
开发语言
xiaoyalian5 小时前
R语言绘图过程中遇到图例的图块中出现字符“a“的解决方法
笔记·r语言·数据可视化