【数据结构初阶】顺序表SeqList

news2025/1/6 20:12:21

描述

顺序表我们可以把它想象成在一个表格里面填数据,并对数据做调整;

那我们的第一个问题是:怎么样在创建出足够的空间呢?

我们可以去堆上申请,用一个指针指向一块空间,如果申请的空间不够,我们可以再realloc申请出来。

我们的第二个问题是:怎么样标记我们用了多少空间呢?

这时我们就需要一个变量来记录我们当前的用到第几个“格子”(即用了多少空间),我们这里用size来表示:

我们的第三个问题是:怎么样知道我们有空间呢?

这时我们就需要一个变量来记录我们“格子”总数(即拥有多少空间),我们这里用capacity来表示:

所以(在.h文件中)我们线性表的结构体描述为:

typedef int SLDataType;

typedef struct SeList
{
	SLDataType* a;
	int size;
	int capacity;
}SL;

组织

  1. 初始化
  2. 释放
  3. 尾插
  4. 尾删
  5. 头插
  6. 头删
  7. 指定位置删除
  8. 指定位置插入
  9. 查找指定元素

1.初始化

把我们描述出来的顺序表结构体里的变量初始化

在.h文件中:

因为要对创建出来的结构体里的内容进行修改,所以函数要进行传址调用

在.c文件中:

我们用malloc来开辟空间,同时注意检查malloc;

因为我们刚刚开辟空间,并没有往顺序表里增删查改 所以此时size为0;

而我们的capacity就是在malloc开辟的空间大小。

void SLInit(SL* ps)
{
	assert(ps->a);

	ps->a = (SLDataType*)malloc(sizeof(SLDataType) * INIT_CAPACITY);
	if (ps->a == NULL)
	{
		perror("malloc fail");
		return;
	}

	ps->size = 0;
	ps->capacity = INIT_CAPACITY;
}

2.释放

将描述出来的顺序表结构体里的变量逐个进行释放;

在.h 文件中:

在.c文件中:

首先是指针a,需要使用free函数将其释放,还需要注意的是free后,需要将a置空,避免出现野指针

因为size和capacity是临时变量储存在栈上,函数调用结束后会自动释放,我们这里把它改为0就可以了。

void SLDestroy(SL* ps)
{
	assert(ps->a);

	free(ps->a);
	ps->a = NULL;
	ps->capacity = 0;
	ps->size = 0;
}

3.尾插

向顺序表的尾部插入数据;

在.h 文件中:

在.c文件中:

尾插数据时,首先,需要判断capacity(空间)是否足够,如果不够需要扩容,这里我们写个扩容函数:

扩容我们用realloc函数,最后要返回ps至尾插函数判断是否为空;

接着,才能将数据尾插;

最后别忘了调整size的位置;

SL* SLExpand(SL* ps)
{
	assert(ps->a);
	if (ps->size == ps->capacity)
	{
		SLDataType* tmp = (SLDataType*)realloc(ps->a, sizeof(SLDataType)* ps->capacity*2);
		if (tmp == NULL)
		{
			perror("realloc fail");
			return NULL;
		}
		ps->capacity *=2;
		ps->a = tmp;
	}
	return ps;
}

void SLPushBack(SL* ps,SLDataType x)
{
	assert(ps->a);
	//扩容
	if (ps->size == ps->capacity)
	{
		SL* ret = SLExpand(ps);
		if (ret == NULL)
		{
			return;
		}
	}
	ps->a[ps->size] = x;
	ps->size++;

}

4.尾删

将顺序表的尾部删除;

在.h 文件中:

在.c 文件中:

因为顺序表是从开始连续存储size个数据,不能单独释放那一块区域,所以我们直接将size--就可以了,如果往后插入的话,就直接把数据覆盖;

assert()如果当capacity为空的时候还尾删时会报错,并且终止程序;

void SLPopBack(SL* ps)
{
	assert(ps->a);
	ps->size--;
}

5.头插

向顺序表的头部插入数据;

在.h 文件中:

在.c 文件中:

首先,我们得先判断空间是否足够,如果不够就扩容;

第二步:把数据往后移一位,数据从最后一位开始向后移动;

第三步:进行数据头插,别忘了把size的大小改改;

void SLPushFront(SL* ps, SLDataType x)
{
	assert(ps->a);

	//扩容
	if (ps->size == ps->capacity)
	{
		SL* ret = SLExpand(ps);
		if (ret == NULL)
		{
			return;
		}
	}

	//移动数据
	for (int i = ps->size; i >0 ; i--)
	{
		ps->a[i] = ps->a[i - 1];
	}

	//头插
	ps->a[0] = x;
	ps->size++;

}

6.头删

将顺序表的头部数据删除;

在.h 文件中:

在.c 文件中:

我们只需要将从第二位数据开始往前移动,把前一位的数据覆盖就可以达到头删的效果;

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

	ps->size--;
}

7.指定位置删除

将顺序表的指定位置数据删除;

在.h 文件中:

在.c 文件中:

首先,要先用assert检查一下空间和size的值是否合理;

如果删除的数据是最后一个就直接尾删;

如果如果删除的数据不是最后一个就需要移动数据覆盖,类似于头删;

void SLErase(SL* ps, int pos)
{
	assert(ps->a);
	assert(pos >= 0&&pos<ps->size);

	//如果pos是最后一个数据,尾删
	if (pos == ps->size - 1)
	{
		SLPopBack(ps);
	}

	else
	{
		for (int i = pos; i < ps->size; i++)
		{
			ps->a[i] = ps->a[i + 1];
		}
		ps->size--;
	}

	
}

8.指定位置插入

将顺序表的指定位置数据插入;

在.h 文件中:

在.c 文件中:

首先,要先用assert检查一下空间和size的值是否合理;

如果删除的数据是最后一个就直接尾插;

如果如果删除的数据不是最后一个就需要移动数据,再插入数据,类似于头插;

void SLInsert(SL* ps, int pos, SLDataType x)
{
	assert(ps->a);
	assert(pos >= 0 && pos <= ps->size);

	//判断容量
	if (ps->size == ps->capacity)
	{
		SL* ret = SLExpand(ps);
		if (ret == NULL)
		{
			return;
		}
	}

	//尾插
	if (pos == ps->size - 1)
	{
		SLPushBack(ps,x);
	}
	else
	{
		for (int i = ps->size; i > pos; i--)
		{
			ps->a[i] = ps->a[i - 1];
		}
		ps->size++;
		ps->a[pos] = x;
	}
}

9.查找指定元素

在顺序表中查找指定数据,并输出其下表,和在该表中的个数;

在.h 文件中:

在.c 文件中:

for循环遍历一下顺序表,如果遍历过程中找到了直接打印其下标,并用一个变量记录它出现的此数。

出循环后打印其和在该表中出现的个数。

void SLFindPoint(SL* ps, SLDataType x)
{
	assert(ps->a);
	
	int cnt = 0;
	for (int i = 0; i < ps->size; i++)
	{
		if (ps->a[i] == x)
		{
			cnt++;
			printf("找到了第%d个,下标为:%d\n",cnt, i);
		}
	}
	if (cnt == 0)
	{
		printf("抱歉,无该数据\n");
	}
	else
	{
		printf("共找到%d个数据\n", cnt);
	}
}

整个程序

.h文件:

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<string.h>

typedef int SLDataType;
#define INIT_CAPACITY 4

typedef struct SeList
{
	SLDataType* a;
	int size;
	int capacity;
}SL;

void SLPrint(SL* ps);

void SLInit(SL* ps);
void SLDestroy(SL* ps);
void SLPushBack(SL* ps,SLDataType x);
void SLPopBack(SL* ps);
void SLPushFront(SL* ps, SLDataType x);
void SLPopFront(SL* ps);
void SLErase(SL* ps,int pos);
void SLInsert(SL* ps, int pos, SLDataType x);
void SLFindPoint(SL* ps, SLDataType x);

.c文件:

#define _CRT_SECURE_NO_WARNINGS 1
#include"Seqlist.h"


void SLPrint(SL* ps)
{
	assert(ps->a);

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


void SLInit(SL* ps)
{
	assert(ps->a);

	ps->a = (SLDataType*)malloc(sizeof(SLDataType) * INIT_CAPACITY);
	if (ps->a == NULL)
	{
		perror("malloc fail");
		return;
	}

	ps->size = 0;
	ps->capacity = INIT_CAPACITY;
}

void SLDestroy(SL* ps)
{
	assert(ps->a);

	free(ps->a);
	ps->a = NULL;
	ps->capacity = 0;
	ps->size = 0;
}

SL* SLExpand(SL* ps)
{
	assert(ps->a);
	if (ps->size == ps->capacity)
	{
		SLDataType* tmp = (SLDataType*)realloc(ps->a, sizeof(SLDataType)* ps->capacity*2);
		if (tmp == NULL)
		{
			perror("realloc fail");
			return NULL;
		}
		ps->capacity *=2;
		ps->a = tmp;
	}
	return ps;
}

void SLPushBack(SL* ps,SLDataType x)
{
	assert(ps->a);
	//扩容
	if (ps->size == ps->capacity)
	{
		SL* ret = SLExpand(ps);
		if (ret == NULL)
		{
			return;
		}
	}
	ps->a[ps->size] = x;
	ps->size++;

}

void SLPopBack(SL* ps)
{
	assert(ps->a);
	ps->size--;
}

void SLPushFront(SL* ps, SLDataType x)
{
	assert(ps->a);

	//扩容
	if (ps->size == ps->capacity)
	{
		SL* ret = SLExpand(ps);
		if (ret == NULL)
		{
			return;
		}
	}

	//移动数据
	for (int i = ps->size; i >0 ; i--)
	{
		ps->a[i] = ps->a[i - 1];
	}

	//头插
	ps->a[0] = x;
	ps->size++;

}

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

	ps->size--;
}

void SLErase(SL* ps, int pos)
{
	assert(ps->a);
	assert(pos >= 0&&pos<ps->size);

	//如果pos是最后一个数据,尾删
	if (pos == ps->size - 1)
	{
		SLPopBack(ps);
	}

	else
	{
		for (int i = pos; i < ps->size; i++)
		{
			ps->a[i] = ps->a[i + 1];
		}
		ps->size--;
	}

	
}

void SLInsert(SL* ps, int pos, SLDataType x)
{
	assert(ps->a);
	assert(pos >= 0 && pos <= ps->size);

	//判断容量
	if (ps->size == ps->capacity)
	{
		SL* ret = SLExpand(ps);
		if (ret == NULL)
		{
			return;
		}
	}

	//尾插
	if (pos == ps->size - 1)
	{
		SLPushBack(ps,x);
	}
	else
	{
		for (int i = ps->size; i > pos; i--)
		{
			ps->a[i] = ps->a[i - 1];
		}
		ps->size++;
		ps->a[pos] = x;
	}
}

void SLFindPoint(SL* ps, SLDataType x)
{
	assert(ps->a);
	
	int cnt = 0;
	for (int i = 0; i < ps->size; i++)
	{
		if (ps->a[i] == x)
		{
			cnt++;
			printf("找到了第%d个,下标为:%d\n",cnt, i);
		}
	}
	if (cnt == 0)
	{
		printf("抱歉,无该数据\n");
	}
	else
	{
		printf("共找到%d个数据\n", cnt);
	}
}

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

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

相关文章

Spring基础学习——web

Spring基础学习——web 一、Spring整合Web环境1.1 JavaWeb三大组件作用及其特点1.2 Spring整合Web环境的思路及实现1.3 Spring开发Web环境组件spring-web1.4 web层MVC框架思想与设计思路 一、Spring整合Web环境 1.1 JavaWeb三大组件作用及其特点 在Java语言当中&#xff0c;w…

2023亚太杯数学建模A题思路

文章目录 0 赛题思路1 竞赛信息2 竞赛时间3 建模常见问题类型3.1 分类问题3.2 优化问题3.3 预测问题3.4 评价问题 4 建模资料5 最后 0 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 1 竞赛信息 2023年第十三…

【Java】Java8 Function 和 Consumer 接口的使用场景

文章目录 前言1. Function 示例2. Function 介绍3. Consumer 示例4. Consumer 介绍5. Function 和 Consumer 接口的使用场景后记 前言 在 《精通Java8》一书中有讲过 Java8的函数式接口可以简化设计模式的实施&#xff0c;这里记录一下Function 和 Consumer 的使用场景。 1. …

docker主备节点数据同步

主备节点挂载 在生产环境中&#xff0c;赋予一个docker操作系统的权限是一件不安全的事&#xff0c;在不具有系统操作权限的情况下&#xff0c;主备机无法通过nfs进行挂载。此时&#xff0c;可借助数据卷进行挂载 创建两个数据卷 docker volume create vol1 docker volume cr…

Java常用设计模式(23种)

文章目录 介绍 设计模式的六大原则 一、创建型模式 1、单例模式&#xff08;Singleton Pattern&#xff09; 1&#xff09;饿汉式 2&#xff09;懒汉式&#xff0c;双检锁 3&#xff09;静态内部类 4&#xff09;枚举 2、原型模式&#xff08;Prototype Pattern&#xff09…

(免费领源码)java#ssm#mysql在线学习平台85204-计算机毕业设计项目选题推荐

摘 要 随着科学技术的飞速发展&#xff0c;社会的方方面面、各行各业都在努力与现代的先进技术接轨&#xff0c;通过科技手段来提高自身的优势&#xff0c;在线学习平台当然也不能排除在外。在线学习平台是以实际运用为开发背景&#xff0c;运用软件工程原理和开发方法&#x…

线索二叉树(存储结构,线索化,寻找前驱/后继)

目录 1.线索二叉树1.中序线索二叉树2.后序线索二叉树3.先序线索二叉树 2.线索二叉树的存储结构3.二叉树的线索化1.中序线索化2.先序线索化3.后序线索化 4.寻找前驱/后继1.中序线索二叉树找后继2.中序线索二叉树找中序前驱3.先序线索二叉树找先序后继4.先序线索二叉树找先序前驱…

会员题-力扣408-有效单词缩写

有效单词缩写 字符串可以用 缩写 进行表示&#xff0c;缩写 的方法是将任意数量的 不相邻 的子字符串替换为相应子串的长度。例如&#xff0c;字符串 “substitution” 可以缩写为&#xff08;不止这几种方法&#xff09;&#xff1a; “s10n” (“s ubstitutio n”) “sub4…

配置交换机将Log发送到日志服务器

文章目录 一、配置说明二、配置步骤推荐阅读 一、配置说明 配置将实现如下&#xff1a; 配置交换机将Log发送到日志服务器。 将信息等级高于等于 debug 的日志信息将会发送到日志服务器上。 允许输出日志信息的模块为default所有应用模块日志信息。 SW1为我们日志源交换机…

Clickhouse学习笔记(12)—— 物化视图

ClickHouse 的物化视图是一种查询结果的持久化&#xff0c;与普通视图对比&#xff0c;其不仅保存了查询的逻辑&#xff0c;还保存了查询结果&#xff1b; 物化视图与普通视图的区别 普通视图不保存数据&#xff0c;保存的仅仅是查询语句&#xff0c;查询的时候还是从原表读取…

画面精美传奇手游幽冥传奇【幽冥灭龙传奇】win服务端+双端+GM授权后台+详细教程

搭建资源下载地址&#xff1a;画面精美传奇手游幽冥传奇幽冥灭龙传奇win服务端双端GM授权后台详细教程-海盗空间

JuCheap开发的微信小程序商城(NetCore商城)

一、目的 最近工作需要&#xff0c;在学习微信小程序的开发&#xff0c;用周末空闲时间开发了一个微信小程序商城。 二、功能 2.1 管理后台 管理后台是基于JuCheap开发的&#xff0c;使用Net6Vue3ElementPlus开发&#xff0c;具体功能包含如下&#xff1a; 2.1.1 店铺模块…

lc228. 汇总区间

暴力解法&#xff1a;遍历数组&#xff0c;判断数组是否连续递增。将连续递增的数据的首尾的数据分别存储在map集合的key和value之中&#xff0c;输出时判断首尾值是否相同采用两种方式输出 复杂度分析 时间复杂度&#xff1a;O(n)&#xff0c;其中 n 为数组的长度。空间复杂…

环形链表解析(c语言)c语言版本!自我解析(看了必会)

目录 1.判断一个表是否是环形链表&#xff01; 代码如下 解析如下 2.快指针的步数和慢指针的步数有什么影响&#xff08;无图解析&#xff09; 3.怎么找到环形链表的入环点 代码如下 解析如下 1.判断一个表是否是环形链表&#xff01; 代码如下 bool hasCycle(struct L…

ZDH-智能营销-执行流程解析

目录 项目源码 预览地址 安装包下载地址 通过2个方向解读ZDH流程图 图执行方向 数据流转方向 项目源码 zdh_web: GitHub - zhaoyachao/zdh_web: 大数据采集,抽取平台 zdh_magic_mirror: https://github.com/zhaoyachao/zdh_magic_mirror 预览地址 后台管理-登陆 用户…

Matplotlib绘图一网打尽【持续更新ing】

2 绘制扇形图 绘制一个展示男女乘客比例的扇形图 得出男女的具体数字 sex_per df["Sex"].value_counts() sex_per # 把画图的包导入进来 import matplotlib.pyplot as plt# 这种绘图方式主要用于有多个子图以及复杂的图形布局的时候。fig,ax plt.subplots()# pl…

numpy 基础使用

NumPy是Python中科学计算的基础包。它是一个Python库&#xff0c;提供多维数组对象&#xff0c;各种派生对象&#xff08;如掩码数组和矩阵&#xff09;&#xff0c;以及用于数组快速操作的各种API&#xff0c;有包括数学、逻辑、形状操作、排序、选择、输入输出、离散傅立叶变…

Python 使用OS模块调用 cmd

嗨喽&#xff0c;大家好呀~这里是爱看美女的茜茜呐 在os模块中提供了两种调用 cmd 的方法&#xff0c;os.popen() 和 os.system() os.system(cmd) 是在执行command命令时需要打开一个终端&#xff0c;并且无法保存command命令的执行结果。 os.popen(cmd,mode) 打开一个与comma…

汽车ECU的虚拟化技术初探(一)

目录 1.为什么要提汽车ECU的虚拟化&#xff1f; 2.虚拟化技术分类 2.1 硬件虚拟化 2.2 操作系统虚拟化 问题引入&#xff1a; Hypervisor是如何来管理和隔离硬件资源&#xff0c;保证各个不同功能的应用程序的资源使用安全和资源调度&#xff1f;没有MMU就做不了虚拟化&am…

【集简云调度影刀RPA】

集简云调度影刀 集简云的http请求&#xff0c;都是用webhook。 1、获取token的时候&#xff0c;在url中必须这么填&#xff0c;在数据或者headers里面填写keyID和密码不管用。 2.调起应用的时候&#xff0c;需要选择webhook中的post&#xff0c;自定义的请求&#xff0c;才能…