一篇博客读懂顺序表 —— Sequence-List

news2025/7/23 6:05:17

目录

一、顺序表的初始定义

1.1新建头文件和源文件

1.2 SeqList.h 中的准备工作

二、顺序表的初始化与销毁

三、首尾插入元素

四、首尾删除元素

五、中间插入元素

六、中间删除元素

 七、查找指定元素下标

八、源代码


一、顺序表的初始定义

1.1新建头文件和源文件

当我们要实现通讯录时,我们会自定义一个 contact.h 文件来存储我们的各种声明,自定义一个 contact.c 文件来存储实现声明的函数,同时还会存在一个 test.c 来测试我们代码的可行性。

在这里我们学习顺序表时,也要使用这种方式,来分割我们的代码使程序更加简洁耐看:

在 SeqList.h 中,我们要声明我们需要的头文件、重新定义的类型、我们需要的函数...... 

1.2 SeqList.h 中的准备工作

为了方便我们修改顺序表中的数据类型,我们把我们顺序表中的类型(int为例)重定义为SLDataType,这样如果我们以后想修改数据类型的话,可以直接来此处将 int 改为 double......

typedef int SLDataType;

 其次,我们定义的顺序表其实是一个结构体,其包含了一个表头(一个指针),实际保存的数据以及表的容量,这里我们并把结构体名称重定义为简称,方便后续的使用。

typedef struct SeqList
{
	int* p;//表头
	int size;//实际存储的数据数量
	int capacity;//此时表中的容量
}SL;

我们的顺序表要具有哪些特点呢?

1.动态存储,可以动态开辟使用空间

2.各种位置的增删查改,分为头、尾、中间。 

二、顺序表的初始化与销毁

初始化和销毁:

首先用断言来保证传入的指针不为空,其次我们需要用 SLInit 函数来将结构体中的数据一一赋初值,其次在销毁数据时,也要保证 free 函数的对象为非空指针,接着将数据重新初始化。

void SLInit(SL* psl) //初始化顺序表
{
	assert(psl != NULL);
	psl->p = NULL;
	psl->size = 0;
	psl->capacity = 0;
}
void SLDestroy(SL* psl) //销毁顺序表
{
	assert(psl != NULL);
	if (psl->p != NULL)
	{
		free(psl->p);
		psl->size = 0;
		psl->capacity = 0;
	}
}

因为我们的顺序表是动态开辟空间,所以写一个检查实时容量的函数是必备的,在此处我们使用二倍扩容的方法来开辟内存空间,但是在初始化时我们把我们的 capacity 赋值为 0,在进行二倍扩容时还是 0,这时就可以用三目运算符完美规避这个问题。

检查并扩充容量:

并且我们用新的临时变量来保存扩容后的空间,在没有问题后再把值返回给原本的变量。

void SLCheckCapacity(SL* psl)
{
	assert(psl != NULL);
	if (psl->size == psl->capacity)
	{
        //因为初始化时capacity为0,所以我们按照二倍扩容后也是0,这里运用三目运算符就能很好的解决
		SLDataType NewCapacity = (psl->capacity == 0) ? 4 : psl->capacity * 2;
        //动态开辟的空间是给顺序表的,注意不要把改行上下两个数据颠倒
        //sizeof() 不要忘!!
		SLDataType* tmp = (SLDataType*)realloc(psl->p, sizeof(SL) * NewCapacity);
		if (tmp == NULL)
		{
			perror("SLCheckCapacity -> realloc");
			return;
		}
		psl->capacity = NewCapacity;
		psl->p = tmp;
	}
}

三、首尾插入元素

打印顺序表:

 为了更好的测试我们的代码,我们可以先写一个打印函数来打印我们的顺序表。

void SLPrint(SL* psl)
{
	assert(psl != NULL);
	int i = 0;
	for (; i < psl->size; i++)
	{
		printf("%d ", psl->p[i]);
	}
	printf("\n");
}

尾部插入元素:

void SLPushBack(SL* psl, SLDataType x)
{
	assert(psl != NULL);
	SLCheckCapacity(psl);//检查是否需要扩容
	psl->p[psl->size] = x;
	psl->size++;
}

首部插入元素:

首部插入元素就比尾部插入元素复杂一点啦,我们需要让前面的元素覆盖后面的元素,下图我们模拟顺序表中有 8 个元素(size == 8),来看一下我们的代码应该如何写:

我们让 i 从后面开始向前走,才能保证有用的元素不会被覆盖,同时我们根据首尾元素的覆盖下标推理出 i 的取值范围。

//第一种取值
void SLPushFront(SL* psl, SLDataType x)
{
	assert(psl != NULL);
	SLCheckCapacity(psl);
	int i = psl->size;
	for (; i > 0; i--)
	{
		psl->p[i] = psl->p[i - 1];
	}
	psl->p[0] = x;
	psl->size++;
}

//第二种取值
void SLPushFront(SL* ps, SLDateType x)//ʱ临Ӷ O(n) ;  n O(n^2)
{
	assert(ps != NULL);
	SLCheckCapacity(ps);
	int i = ps->size - 1;
	for (; i >= 0; i--)
	{
		ps->p[i + 1] = ps->p[i];
	}
	ps->p[0] = x;
	ps->size++;
}

四、首尾删除元素

尾部删除元素:

这里我们采用 size-- 的方法,让我们直接无法访问到最后一个元素,下一次增添时又会被新的元素覆盖以实现删除的操作,同时断言我们的实际元素个数必须多余 0

void SLPopBack(SL* psl)
{
	assert(psl != NULL);
    assert(psl->size > 0);
	psl->size--;
}

首部删除元素: 

void SLPopFront(SL* psl)
{
	assert(psl != NULL);
	int i = 1;
	for (; i <= psl->size; i++)
	{
		psl->p[i - 1] = psl->p[i];
	}
	psl->size--;
}

void SLPopFront(SL* ps)
{
	assert(ps != NULL);
	assert(ps->size > 0);
	int i = 0;
	for (; i < ps->size - 1; i++)
	{
		ps->p[i] = ps->p[i + 1];
	}
	ps->size--;
}

五、中间插入元素

void SLInsert(SL* psl, int num, SLDataType x)
{
	assert(psl != NULL);
	assert(num >= 0 && num <= psl->size);
	SLCheckCapacity(psl);
	int i = psl->size - 1;
	for (; i >= num; i--)
	{
		psl->p[i + 1] = psl->p[i];
	}
	psl->p[num] = x;
	psl->size++;
}

六、中间删除元素

void SLErase(SL* psl, int num)
{
	assert(psl != NULL);
	assert(num >= 0 && num < psl->size);
	SLCheckCapacity(psl);
	int i = num;
	for (; i < psl->size - 1; i++)
	{
		psl->p[i] = psl->p[i + 1];
	}
	psl->size--;
}

 七、查找指定元素下标

int SLFind(SL* ps, SLDataType x)
{
	assert(ps != NULL);
	int i = 0;
	for (; i < ps->size; i++)
	{
		if (ps->p[i] == x)
		{
			return i;
		}
	}
	return -1;
}

八、源代码

欢迎光临我的Gitee - Gitee.comicon-default.png?t=N7T8https://gitee.com/bright-and-sparkling-at-night/studying/commit/dd1f9978f81f9decce01805623b4708b7671f3e0

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1157015.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

zip2john 爆破zip工具

1.使用zip2john工具输出zip文件hash Zip2john 1.zip > 1.txt 2.使用john工具进行zip文件爆破 John 1.txt 如图所示&#xff1a; 该zip密码为19950101

面向Three.js开发者的3D自动纹理化开发包

DreamTexture.js 是面向 three.js 开发者的 3D 模型纹理自动生成与设置开发包&#xff0c;可以为 webGL 应用增加 3D 模型的快速自动纹理化能力。 图一为原始模型, 图二图三为贴图后的模型。提示词&#xff1a; city, Realistic , cinematic , Front view ,Game scene graph 1、…

鸿蒙实现类似于安卓的Handler,鸿蒙的延迟和异步操作

在安卓开发的时候我们想要去延迟处理一些事情可以通过 Handler 解决&#xff0c;比如一秒后处理某些事情&#xff1a; new Handler().postDelayed(new Runnable() {Overridepublic void run() {} },1000);鸿蒙也提供了类似的方法&#xff0c;如下: getUITaskDispatcher().del…

C#,数值计算——积分方程与逆理论Fred2的计算方法与源程序

1 文本格式 using System; namespace Legalsoft.Truffer { public abstract class Fred2 { public readonly double a; public readonly double b; public readonly int n; public double[] t; public double[] f; …

ABBYY16最新pdf文档转换图片、Word、Excel和PPT工具

欢迎来到 ABBYY 软件下载中心&#xff0c;FineReader PDF使专业人员能够最大程度地提高数字工作场所的效率。帮助信息工作者轻松地在同一工作流程中对各种文档进行数字化、检索、编辑、加密、共享和协作。 可以对pdf文档进行各种格式转换&#xff0c;任意转换为图片、Word、Ex…

代码版本控制工具GitLab :从安装到使用一步到位

一、GitLab 是什么&#xff1f; 如果听说过 Git 或者 GitHub&#xff0c;那么 GitLab 你一定也听说过。GitLab 是一个用于仓库管理系统的开源项目&#xff0c;使用 Git 作为代码管理工具&#xff0c;并在此基础上搭建起来的 Web 服务。简单理解&#xff1a;GitLab 类似私人版 …

Linux高性能服务器编程——ch12笔记

第12章 高性能I/O框架库Libevent 12.1 I/O框架库概述 基于Reactor模式的I/O框架库包含&#xff1a; 1&#xff09;句柄&#xff1a;与事件源绑定&#xff08;I/O事件、信号和定时事件&#xff09;。当内核检测到就绪事件时&#xff0c;通过句柄来通知应用程序这一事件。I/O事件…

多线程---线程安全的集合类

文章目录 多线程环境使用数组synchronizedListCopyOnWriteArrayList 多线程环境使用队列多线程环境使用哈希表HashTableConcurrentHashMapHashTable VS HashMap VS ConcurrentHashMap 我们原先学习过的集合类&#xff0c;大部分都是线程不安全的。只有Vector、Stack、HashTable…

智能井盖传感器有哪些?万宾科技智能井盖效果

在城市治理过程之中&#xff0c;小小的井盖却成为影响民生的一个重要方面&#xff0c;因为井盖一旦出现问题&#xff0c;会严重影响市民的生命安全&#xff0c;并且传统的井盖一般都会采用人工巡检的方式&#xff0c;这就导致了巡检的难度较大&#xff0c;从而不能对城市各个角…

Hdmi接口与XAPP460

之前写过VGA&#xff0c;其实HDMI也写过&#xff0c;但是没记笔记。 VGA时序 VGA Signal Timing (tinyvga.com) 介绍 规范&#xff1a; 小结&#xff1a; VGA像素编码以后通过差分对端口以十倍像素时钟的速率地串行发送。 官方例程 HDMI编码器 来自xaap460的编码器 // //…

Python Django 之模板继承详解(extends)

文章目录 1 概述1.1 目的1.2 标签&#xff1a;block、extends1.3 目录结构 2 templates 目录2.1 base.html&#xff1a;父页面2.2 login.html&#xff1a;子页面 3 其它代码3.1 settings.py3.2 views.py3.3 urls.py 1 概述 1.1 目的 模板继承 和 类继承 的目的是一样的&#…

机器人仿真-gazebo学习笔记(1)前期准备

1.更新gazebo到gazebo11 1)查看我们的gazebo插件&#xff08;我这里已经是gazebo11了&#xff0c;但对于ubuntu18.04的同学来说应该是gazebo9&#xff09; dpkg -l | grep gazebo 2)卸载gazebo9的插件 sudo apt-get purge gazebo9 gazebo9-common gazebo9-plugin-base libga…

【Java 进阶篇】Java Response 路径详解

在Java Web开发中&#xff0c;处理HTTP响应的路径是一个重要的概念。了解如何正确处理和管理路径对于构建健壮的Web应用程序至关重要。本篇博客将详细介绍Java中的HTTP响应路径&#xff0c;包括路径的组成、相对路径和绝对路径的区别、如何构建和处理路径&#xff0c;以及路径在…

MySQL基础教程下

MySQL 约束 约束是作用于表中字段上的规则&#xff0c;用于限制存储在表中的数据。 目的&#xff1a;保证数据库中数据的正确、有效性和完整性。 约束描述关键字非空不能为nullnot null唯一约束唯一unique主键非空唯一primary key默认默认值default检查约束满足一个条件chec…

铭控传感数字温度变送器丨远传温度变送器在工业中的助您精准测量

秋季的森林被染成了彩色的&#xff0c;地上满是落叶和一些颗粒饱满的果实。一说起栗子&#xff0c;最令人念念不忘的当属刚出锅的糖炒栗子&#xff0c;栗子的外壳在糖类与高温作用下一点点发生非酶褐变&#xff0c;偶尔有栗子外壳破裂的声音&#xff0c;听着心都跟着颤了一下。…

SIT3088E3.0V~5.5V 供电,ESD 15kV HBM,256 节点,14Mbps 半双工 RS485/RS422 收发器

SIT3088E 是一款 3.0V~5.5V 宽电源供电、总线端口 ESD 保护能力 HBM 达到 15kV 以上、总 线耐压范围达到 15V 、半双工、低功耗&#xff0c;功能完全满足 TIA/EIA-485 标准要求的 RS-485 收发器。 SIT3088E 包括一个驱动器和一个接收器&#xff0c;两者均可独立…

NeurIPS 2023 Spotlight | 探索不变学习中的充分必要因果

©PaperWeekly 原创 作者 | 杨梦月 单位 | 伦敦大学学院 研究方向 | 因果推断 NeurIPS 2023 接收的 Spotlight 论文“Invariant Learning via Sufficient and Necessary Cause”提出了一种在不变学习当中学习充分必要因果的解决方案。其能够在从数据中寻找不变表征的基础上…

阿里云服务器ECS经济型e实例2核2G3M带宽优惠价格99元一年

2023阿里云服务器双11优惠价格99元一年经济型e实例&#xff0c;并且续费不涨价&#xff0c;云服务器ECS-经济型e实例2核2G配置、3M带宽、40G ESSD entry系统盘优惠价99元一年&#xff0c;原价956.64元/年&#xff0c;可用于中小型网站建设、开发测试、小程序或app搭建&#xff…

TRS 2023 | Spurs in Millimeter-Wave FMCW Radar System-on-Chip

原始笔记链接: https://mp.weixin.qq.com/s?__bizMzg4MjgxMjgyMg&mid2247486779&idx1&snc75171844595150abc48d2dd59e7255f&chksmcf51bfc2f82636d4e3f9c8526d0f25df1bea5496d9945d1e963961fea8c8fd630e7670b99afb#rd ↑ \uparrow ↑ 打开上述链接即可阅读全文…

文件操作---IO

文章目录 什么是文件&#xff1f;文件的路径文件的系统操作文件的内容操作面向字节流面向字节流的读操作&#xff08;InputStream&#xff09;面向字节流的写操作&#xff08;OutputStream&#xff09; 面向字符流面向字符流的读操作&#xff08;Reader&#xff09;面向字符流的…