利用PHP实现递归删除链表元素的方法示例


Posted in Javascript onOctober 23, 2020

前言

这篇文章介绍一下 递归,递归的本质是将原来的问题转化为更小的同一个问题,解决这些更小问题的过程。下面通过两个递归的例子帮助学习对递归的理解。

1.递归数组求和

例如某个数组 $arr = [1,2,3,4,5,6,7,8,9,10]; 需要求和,通过实现递归函数对数组求和来帮助学习对递归的理解。

1.1 输出文件 output_recursion.php

<?php
require 'ArrayRecursion.php';
/**
 * 递归实现数组求和
 */
$arr = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
echo ArrayRecursion::recursionSum($arr);

1.2 ArrayRecursion 类

这是一个实现数组递归求和的代码,其中 recursionSum() 是一个递归函数,相当于把求和过程转化为更小的求和,最终实现想要的结果:

<?php
/**
 * 使用递归对数组求和 方便对递归的理解
 * Class ArrayRecursion
 */
class ArrayRecursion
{
  public static function sum(array $arr) {
    return self::recursionSum($arr);
  }
  public static function recursionSum(array $arr, $i = 0) {
    if (count($arr) == $i) {
      return 0;
    }
    return $arr[$i] + self::recursionSum($arr, $i + 1);
  }
}

Tips:这个求和过程仅仅只是帮助学习递归思想,实际求和可以直接遍历数组。

2.递归删除链表某个元素

例如某个链表 10->9->8->99->7->99->6->5->99->4->3->2->1->null 需要删除其中值等于 99 的元素,可以通过实现递归来得到删除指定元素的效果。

2.1 输出文件 output_recursion.php

<?php
require 'LinkedList.php';
require 'LinkedListRecursion.php';
/**
 * 首先实例化一个链表,向链表中添加50个元素
 */
$linkedList = new LinkedList();
for ($i = 0; $i < 50; $i++) {
  if ($i % 7 == 0) {
    $linkedList->addFirst(99);
  } else {
    $linkedList->addFirst($i);
  }
}
echo $linkedList->toString();
/**打印链表中元素
 * 99->48->47->46->45->44->43->99->41->40->39->
 * 38->37->36->99->34->33->32->31->30->29->99->27->
 * 26->25->24->23->22->99->20->19->18->17->16->15->
 * 99->13->12->11->10->9->8->99->6->5->4->3->2->1->99->null
 */
//将链表对象传入一个能删除指定元素的方法,如 99
echo LinkedListRecursion::deleteElement($linkedList, 99)->toString();
/**打印
 * 48->47->46->45->44->43->41->40->
 * 39->38->37->36->34->33->32->31->
 * 30->29->27->26->25->24->23->22->
 * 20->19->18->17->16->15->13->12->
 * 11->10->9->8->6->5->4->3->2->1->null
 */

2.2 LinkedList & Node 链表类

这是一个链表类,可以使用 addFirst() 方法向链表头部添加元素,可使用 getHead() 获取链表 head 节点对象信息,可以使用 setHead() 改变 head,另外下面定义了一个链表节点类 Node:

<?php
/**
 * 链表的实现
 * Class LinkedList
 */
class LinkedList
{
  private $dummyHead;
  private $size;
  /**
   * 初始化链表 null->null
   * LinkedList constructor.
   */
  public function __construct() {
    $this->dummyHead = new Node(null, null);
    $this->size = 0;
  }
  /**
   * 获取链表大小
   * @return int
   */
  public function getSize(): int {
    return $this->size;
  }
  /**
   * 判断链表是否为空
   * @return bool
   */
  public function isEmpty(): bool {
    return $this->size == 0;
  }
  /**
   * 在链表的第 index 位置添加元素
   * @param int $index
   * @param $e
   */
  public function add(int $index, $e): void {
    if ($index < 0 || $index > $this->size) {
      echo "索引范围错误";
      exit;
    }
    $prve = $this->dummyHead;
    for ($i = 0; $i < $index; $i++) {
      $prve = $prve->next;
    }
    //将上插入位置的上一个位置的 next 节点指向插入节点,插入节点的 next 节点信息指向原上节点的 next 节点
    $prve->next = new Node($e, $prve->next);
    $this->size++;
  }
  /**
   * 向链表开头添加元素
   * @param $e
   */
  public function addFirst($e): void {
    $this->add(0, $e);
  }
  /**
   * 向链表末尾添加元素
   * @param $e
   */
  public function addLast($e): void {
    $this->add($this->size, $e);
  }
  /**
   * 获取链表第 index 位置元素
   * @param $index
   */
  public function get($index) {
    if ($index < 0 || $index > $this->size) {
      echo "索引范围错误";
      exit;
    }
    $node = $this->dummyHead;
    for ($i = 0; $i < $index + 1; $i++) {
      $node = $node->next;
    }
    return $node->e;
  }
  /**
   * 获取链表第一个元素
   * @return mixed
   */
  public function getFirst() {
    return $this->get(0);
  }
  /**
   * 获取链表最后一个元素
   * @return mixed
   */
  public function getLast() {
    return $this->get($this->size - 1);
  }
  /**
   * 修改链表中第 index 位置元素值
   * @param $index
   * @param $e
   */
  public function update($index, $e) {
    if ($index < 0 || $index > $this->size) {
      echo "索引范围错误";
      exit;
    }
    $node = $this->dummyHead;
    for ($i = 0; $i < $index + 1; $i++) {
      $node = $node->next;
    }
    $node->e = $e;
  }
  /**
   * 判断链表中是否存在某个元素
   * @param $e
   * @return bool
   */
  public function contains($e): bool {
    for ($node = $this->dummyHead->next; $node != null; $node = $node->next) {
      if ($node->e == $e) {
        return true;
      }
    }
    return true;
  }
  /**
   * 删除链表中第 index 位置元素
   * @param $index
   */
  public function remove($index) {
    if ($index < 0 || $index > $this->size) {
      echo "索引范围错误";
      exit;
    }
    if ($this->size == 0) {
      echo "链表已经是空";
      exit;
    }
    $prve = $this->dummyHead;
    for ($i = 0; $i < $index; $i++) {
      $prve = $prve->next;
    }
    $node = $prve->next;
    $prve->next = $node->next;
    $this->size--;
    return $node->e;
  }
  /**
   * 删除链表头元素
   */
  public function removeFirst() {
    return $this->remove(0);
  }
  /**
   * 删除链表末尾元素
   */
  public function removeLast() {
    return $this->remove($this->size - 1);
  }
  /**
   * 获取头结点信息
   * @return mixed
   */
  public function getHead() {
    return $this->dummyHead->next;
  }
  /**
   * 设置头
   * @param Node $head
   */
  public function setHead(Node $head) {
    $this->dummyHead->next = $head;
  }
  /**
   * 链表元素转化为字符串显示
   * @return string
   */
  public function toString(): string {
    $str = "";
    for ($node = $this->dummyHead->next; $node != null; $node = $node->next) {
      $str .= $node->e . "->";
    }
    return $str . "null";
  }
}
class Node
{
  public $e;//节点元素
  public $next; //下个节点信息
  /**
   * 构造函数 设置节点信息
   * Node constructor.
   * @param $e
   * @param $next
   */
  public function __construct($e, $next) {
    $this->e = $e;
    $this->next = $next;
  }
}

2.3 LinkedListRecursion 类

这个类定义了一个 deleteElement(LinkedList $linkedList, $val) 方法可以将传进的链表类中指定元素值的节点删除掉(实际是节点的 next 重新指向),recursionDelete($head, $val) 方法是一个递归函数,它能递归删除 head 中指定元素值等于 $val 的节点删除:

<?php
/**
 * 递归删除链表指定元素
 * Class LinkedListRecursion
 */
class LinkedListRecursion
{
  public static function deleteElement(LinkedList $linkedList, $val) {
    $linkedList->setHead(self::recursionDelete($linkedList->getHead(), $val));
    return $linkedList;
  }
  
   /**
   * 递归函数 递归删除链表元素
   * @param $head
   * @param $val
   * @return null
   */
  private static function recursionDelete($head, $val) {
    if ($head == null) {
      return null;
    } else {
      if ($head->e == $val) {
        return self::recursionDelete($head->next, $val);
      } else {
        $head->next = self::recursionDelete($head->next, $val);
        return $head;
      }
    }
  }
}

代码仓库 :https://gitee.com/love-for-po...

总结

到此这篇关于利用PHP实现递归删除链表元素的文章就介绍到这了,更多相关PHP递归删除链表元素内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
密码强度检测效果实现原理与代码
Jan 04 Javascript
extJS中常用的4种Ajax异步提交方式
Mar 07 Javascript
在JavaScript应用中实现延迟加载的方法
Jun 25 Javascript
全面解析JavaScript中“&amp;&amp;”和“||”操作符(总结篇)
Jul 18 Javascript
Three.js学习之文字形状及自定义形状
Aug 01 Javascript
react-redux中connect()方法详细解析
May 27 Javascript
基于jQuery选择器之表单对象属性筛选选择器的实例
Sep 19 jQuery
JS实现获取自定义属性data值的方法示例
Dec 19 Javascript
在vue使用clipboard.js进行一键复制文本的实现示例
Jan 15 Javascript
bootstrap table列和表头对不齐的解决方法
Jul 19 Javascript
javascript实现评分功能
Jun 24 Javascript
微信小程序实现分页加载效果
Nov 19 Javascript
如何在面试中手写出javascript节流和防抖函数
Oct 22 #Javascript
如何通过Proxy实现JSBridge模块化封装
Oct 22 #Javascript
微信小程序canvas动态时钟
Oct 22 #Javascript
vue-cli —— 如何局部修改Element样式
Oct 22 #Javascript
微信小程序入门之绘制时钟
Oct 22 #Javascript
微信小程序入门之指南针
Oct 22 #Javascript
微信小程序实现拼图小游戏
Oct 22 #Javascript
You might like
星际争霸任务指南——人族
2020/03/04 星际争霸
php调用新浪短链接API的方法
2014/11/08 PHP
PHP获取音频文件的相关信息
2015/06/22 PHP
PHP设计模式之状态模式定义与用法详解
2018/04/02 PHP
PHP echo()函数讲解
2019/02/15 PHP
php面向对象程序设计中self与static的区别分析
2019/05/21 PHP
详解PHP 7.4 中数组延展操作符语法知识点
2019/07/19 PHP
PHP创建XML的方法示例【基于DOMDocument类及SimpleXMLElement类】
2019/09/10 PHP
javascript 冒泡排序 正序和倒序实现代码
2010/12/14 Javascript
基于jquery的下拉框改变动态添加和删除表格实现代码
2020/09/12 Javascript
整理8个很棒的 jQuery 倒计时插件和教程
2011/12/12 Javascript
Javascript this 的一些学习总结
2012/08/02 Javascript
js对文章内容进行分页示例代码
2014/03/05 Javascript
JavaScript数据类型之基本类型和引用类型的值
2015/04/01 Javascript
使用C++为node.js写扩展模块
2015/04/22 Javascript
纯js实现无限空间大小的本地存储
2015/06/18 Javascript
JavaScript中的Promise使用详解
2015/06/24 Javascript
js 获取当前web应用的上下文路径实现方法
2016/08/19 Javascript
AngularJS 与百度地图的结合实例
2016/10/20 Javascript
JS实现滑动门效果的方法详解
2016/12/19 Javascript
详解angular中如何监控dom渲染完毕
2017/01/03 Javascript
JavaScript之DOM_动力节点Java学院整理
2017/07/03 Javascript
select自定义小三角样式代码(实用总结)
2017/08/18 Javascript
JS实现移动端整屏滑动的实例代码
2017/11/10 Javascript
vue axios请求拦截实例代码
2018/03/29 Javascript
element-ui table组件如何使用render属性的实现
2019/11/04 Javascript
[03:15]DOTA2-DPC中国联赛1月22日Recap集锦
2021/03/11 DOTA
python3中的eval和exec的区别与联系
2019/10/10 Python
Python脚本操作Excel实现批量替换功能
2019/11/20 Python
解决python脚本中error: unrecognized arguments: True错误
2020/04/20 Python
英国休闲奢华的缩影:Crew Clothing
2019/05/05 全球购物
外贸实习生自荐信范文
2013/11/24 职场文书
学习十八大报告感言
2014/02/04 职场文书
党政领导班子四风问题对照检查材料思想汇报
2014/10/02 职场文书
书法社团活动总结
2015/05/07 职场文书
面试被问select......for update会锁表还是锁行
2021/11/11 MySQL