【PHP数据结构】顺序表(数组)的相关逻辑操作
在定义好了物理结构,也就是存储结构之后,我们就需要对这个存储结构进行一系列的逻辑操作。在这里,我们就从顺序表入手,因为这个结构非常简单,就是我们最常用的数组。那么针对数组,我们通常都会有哪些操作呢?
不用想得太复杂,我们只需要这几个简单的操作就可以了:
1.查找
2.插入
3.删除
是不是很简单?为什么没有遍历呢?我们经常要去遍历一个数组呀?
请注意,在这里,我们是以数据结构的角度来讲顺序表这个物理结构。遍历操作一般针对的会是更复杂的一些结构,比如树、图,从一个结点开始去遍历所有的路径之类的。而对于顺序表这个物理结构来说来说,我们只需要掌握上述那三个操作,不需要包含遍历。
又有同学说了,在 PHP 中,这三个操作简直太简单好不好,完全没有技术含量呀!
小心不要入坑了哦,查找我们说的是找到这个值所在的下标,而不是给你一个下标简单的输出一个值。另外,插入和删除我们是需要考虑一个问题的,那就是我们第 i 个位置插入或者删除数据之后,i+1 及其之后的数据是不是也要相应的移动呢?要小心,我们是插入和删除一个下标位置的内容,而不是修改替换这个下标的内容!!!
好吧,还是直接以实例来说明。
插入
/**
* 数组插入
* @param array $list 顺序表数组
* @param int $i 插入数据下标
* @param mixed $e 数组元素
* return bool 成功失败结果
*/
function ListInsert(array &$list, int $i, $e)
{
$c = count($list);
if ($i < 0 || $i > $c) {
return false;
}
$j = $c - 1;
while ($j >= $i) {
// 从后往前,下一个位置的值变成现在这个位置的值
// 数据向后挪动
$list[$j + 1] = $list[$j];
$j--;
}
// 在指定位置插入值
$list[$i] = $e;
return true;
}
插入操作首先要判断是否下标越界。接下来就从后往前地将插入位置之后的数据向后挪动一位,最后将新增加的数据放到指定的位置。需要注意的是,在这个操作中,我们最主要关心的就是这个数据位置的移动。我们为什么要从数组最后一位开始进行挪动,而不是从插入位置开始移动呢?如果从插入位置开始,那么后面的数据就会都是一个数据了,也就是插入位置的下一个数据。大家有兴趣的可以自己尝试一下。
$arr = [1, 2, 3, 4, 5, 6, 7];
ListInsert($arr, 3, 55);
print_r($arr);
// Array
// (
// [0] => 1
// [1] => 2
// [2] => 3
// [3] => 55
// [4] => 4
// [5] => 5
// [6] => 6
// [7] => 7
// )
在上面的测试代码中,我们往数据的位置 3 处插入一个数据 55 。可以看到输出的结果,数组长度增加了一位,并且从下标 3 的位置开始,后面的数据都向后移动了一位。
删除
/**
* 删除指定下标元素
* @param array $list 顺序表数组
* @param int $i 插入数据下标
* return bool 成功失败结果
*/
function ListDelete(array &$list, int $i)
{
$c = count($list);
if ($i < 0 || $i > $c - 1) {
return false;
}
$j = $i;
while ($j < $c) {
// 当前位置的值变成下一个位置的值
// 数据向前挪动
$list[$j] = $list[$j+1];
$j++;
}
// 去掉最后一个数据
unset($list[$c - 1]);
return true;
}
学习了上面的插入操作之后,相信大部分同学也能想象到删除元素的操作正好跟插入是返过来的。第一步依然还是判断下标是否合规。接下来就是把指定删除的下标元素之后的元素向前挪动一位。在这里,我们是从删除下标开始将元素依次向前移动一位,最后再删除掉重复的最后一位数据,也就是实现数组元素数量的减 1 操作。
$arr = [1, 2, 3, 4, 5, 6, 7];
ListDelete($arr, 5);
print_r($arr);
// Array
// (
// [0] => 1
// [1] => 2
// [2] => 3
// [3] => 4
// [4] => 5
// [5] => 7
// )
测试结果也很清楚,原来在下标 5 位置的元素是 6 。我们删除了下标为 5 的元素后,整个数据的元素数量减少了一位,后面的元素要移动上来,也就是元素 7 要移动到 5 的位置上来。
查找
查找就是简单的做一个线性查找即可,也就是一个一个的去比对数据,看我们需要的数据在数组的哪个位置。
/**
* 查找
* @param array $list 顺序表数组
* @param mixed $e 数组元素
* return int 查找结果下标
*/
function LocateElem(array $list, $e)
{
$c = count($list);
for ($i = 0; $i < $c; $i++) {
if ($list[$i] == $e) {
return $i;
}
}
return -1;
}
如果找到了数据,我们就返回当前数据所在位置的下标。如果到最后依然没有找到对应的数据,就返回一个 -1 表示我们没有找到对应的数据。
总结
欢迎进入数据结构与算法的世界,意不意外,惊不惊喜,今天第一次写这么多代码,但是写出来的是不是感觉和我们平常写的不太一样?就像插入和删除的数据移动一样,如果平常没注意的话可能还真的不知道我们应该反过来移动才能得到正确的结果。这就是数据结构和算法学习的乐趣,挑战自己,每一天都是超越!
测试代码:
参考资料:
《数据结构》第二版,严蔚敏
《数据结构》第二版,陈越
《数据结构高分笔记》2020版,天勤考研
【PHP数据结构】顺序表(数组)的相关逻辑操作的更多相关文章
- c数据结构 顺序表和链表 相关操作
编译器:vs2013 内容: #include "stdafx.h"#include<stdio.h>#include<malloc.h>#include& ...
- hrbust-1545-基础数据结构——顺序表(2)
http://acm.hrbust.edu.cn/index.php?m=ProblemSet&a=showProblem&problem_id=1545 基础数据结构——顺序表(2) ...
- hrbustoj 1545:基础数据结构——顺序表(2)(数据结构,顺序表的实现及基本操作,入门题)
基础数据结构——顺序表(2) Time Limit: 1000 MS Memory Limit: 10240 K Total Submit: 355(143 users) Total Accep ...
- 数据结构---顺序表(C++)
顺序表 是用一段地址连续的存储单元依次存储线性表的数据元素. 通常用一维数组来实现 基本操作: 初始化 销毁 求长 按位查找 按值查找 插入元素 删除位置i的元素 判空操作 遍历操作 示例代码: // ...
- 数据结构顺序表Java实现
Java实现顺序表算法:1:首先我们需要定义我们的接口,关于顺序表的一些基本的操作:顺序表中的操作都有增删改查. //List接口 public interface IList { //返回线性表的大 ...
- python算法与数据结构-顺序表(37)
1.顺序表介绍 顺序表是最简单的一种线性结构,逻辑上相邻的数据在计算机内的存储位置也是相邻的,可以快速定位第几个元素,中间不允许有空,所以插入.删除时需要移动大量元素.顺序表可以分配一段连续的存储空间 ...
- Java数据结构——顺序表
一个线性表是由n(n≥0)个数据元素所构成的有限序列. 线性表逻辑地表示为:(a0,a1,…,an-1).其中,n为线性表的长度,n=0时为空表.i为ai在线性表中的位序号. 存储结构:1.顺序存储, ...
- 数据结构顺序表删除所有特定元素x
顺序表类定义: template<class T> class SeqList : { public: SeqList(int mSize); ~SeqList() { delete[] ...
- 数据结构——顺序表(sequence list)
/* sequenceList.c */ /* 顺序表 */ /* 线性表的顺序存储是指在内存中用地址连续的一块存储空间顺序存放线性表中的各项数据元素,用这种存储形式的线性表称为顺序表. */ #in ...
- 数据结构顺序表中Sqlist *L,&L,Sqlist *&L
//定义顺序表L的结构体 typedef struct { Elemtype data[MaxSize]: int length; }SqList; //建立顺序表 void CreateList(S ...
随机推荐
- Notes about "Exploring Expect"
Chapter 3 Section "The expect Command": expect_out(0,string) can NOT be written as "e ...
- [TcaplusDB知识库]数据库支撑底盘引擎计算层介绍
在上次的TcaplusDB知识库中,TcaplusDB君为大家讲解了TcaplusDB所用的基于HASH表的Key-value存储引擎TXHDB.存储引擎作为数据库的支撑底盘,其重要性无可置疑,而在本 ...
- [ES6深度解析]12:Classes
我们将讨论一个老问题:在JavaScript中创建对象的构造函数. 存在的问题 假设我们想要创建最典型的面向对象设计的示例:Circle类.假设我们正在为一个简单的Canvas库编写一个Circle. ...
- idea中使用docker插件部署项目
安装docker 如果你之前安装过 docker,请先删掉 sudo yum remove docker docker-common docker-selinux docker-engine 安装一些 ...
- 上传jar包到nexus
注释掉: org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.meeno.boot.oa.OaAutoConfigur ...
- 基于css的一些动画
最近因为期末复习周,博客更新鸽了很久,趁着考完试还记得这件事,把之前的大作业里出现过的css动画总结一下 页脚的联系方式图标 这个图片原型是一个静态图 动画效果如下 html <div clas ...
- C++ native 转 C++的CLI
- 图解Win 10 应用开发之Sqlite 数据库的简单用法
尽管目前 UWP-RT 库中还没有自带操作Sqlite数据库的API,不过,真要使用的话也不难,因为通过 Nuget ,我们其实可以获取很多支持 Sqlite 操作的第三方组件,当然了,组件虽多,但不 ...
- C#异步编程 Task await的理解
async/await是C#5.0中推出的,先上用法: static void Main(string[] args) { Console.WriteLine("-------主线程启动-- ...
- java多线程的一些面试题
8.callable与fature Callable与Runnable类似,但是Callable有返回值,并且有一个参数化的类型. Fature保存异步计算的结果.9.执行器 Executor.10. ...