一起刷题(leetcode)第二篇:如何用Python实现递归
时间:2022-07-22
本文章向大家介绍一起刷题(leetcode)第二篇:如何用Python实现递归,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
01
前言
我们知道递归是一类比较巧妙但是理解难度有点大的算法,对于工作中需要用到数据结构和高级算法的人需要牢固掌握递归算法。今天就以实际的案例来带大家一起学习和理解如何用Python实现递归算法。
02
升序列表合并
题目:
将两个升序链表合并为一个新的 升序 链表并返回。
新链表是通过拼接给定的两个链表的所有节点组成的。
示例:
输入:1->2->4, 1->3->4
输出:1->1->2->3->4->4
对于这一道关于链表的问题,由于要求新的链表是通过改变原来两个链表,所以我们不可以新建一个链表来搭建新的链表而是在原来链表基础上改变链表节点中的指向。
我们发现,合并两个链表可以拆解成合并一个链表的指向以及链表的一个节点。由于原问题和子问题具有相同的结构,因此我们可以考虑使用自上而下的递归来解决:
class Solution:
def mergeTwoLists(self, l1, l2):
if l1 is None:
return l2
elif l2 is None:
return l1
elif l1.val < l2.val:
l1.next = self.mergeTwoLists(l1.next, l2)
return l1
else:
l2.next = self.mergeTwoLists(l1, l2.next)
return l2
值得注意的是,这里的mergeTwolists函数在类中,调用的话需要在前面加上self。可以代码看出来递归写起来形式非常简单。
03
对称二叉树
题目:
给定一个二叉树,检查它是否是镜像对称的。
例如,二叉树 [1,2,2,3,4,4,3] 是对称的。
1
/
2 2
/ /
3 4 4 3
但是下面这个 [1,2,2,null,3,null,3] 则不是镜像对称的:
1
/
2 2
3 3
对于二叉树,我们其实有两种遍历方法:广度优先搜索以及深度优先搜索。从字面的意思就可以理解,广度优先是一层层搜索,需要用到“队列”这种数据结构。而深度优先是一条道走到黑,走到最深处后再搜索另一条路径,可以用递归的方法来完成:
class Solution:
def isSymmetric(self, root: TreeNode) -> bool:
if not root:
return True
else:
return self.isSymmetricTree(root.left,
root.right)
def isSymmetricTree(self, left, right):
if left is None and right is None: return True #同时为空
if left is None or right is None : return False #一个为空
if left.val != right.val : return False # 值不相等 !!
return self.isSymmetricTree(left.left, right.right)
and self.isSymmetricTree(left.right, right.left)
在第一个函数isSymmetric中,我们考虑到一种特殊情况就是树为空,同时由于题目给定的函数输入只有一个节点,不适合作为递归函数,因此我们需要额外定义一个新的函数isSymmetricTree来实现递归。
04
二叉树的最大深度
题目:
给定一个二叉树,找出其最大深度。
二叉树的深度为根节点到最远叶子节点的最长路径上的节点数。
说明: 叶子节点是指没有子节点的节点。
示例:
给定二叉树 [3,9,20,null,null,15,7],
3
/
9 20
/
15 7
返回它的最大深度 3 。
这道题和上面对称二叉树都是针对二叉树的,因此也适合用递归实现深度优先搜索来计算二叉树的最大深度:
class Solution:
def maxDepth(self, root: TreeNode) -> int:
if not root:
return 0
if not root.left and not root.right:
return 1
left = self.maxDepth(root.left)
right = self.maxDepth(root.right)
return 1+max(left,right)
这里我们本身定义的函数的输入是一个节点,可以用来作为递归函数。
- readlink: command not found 解决方案
- Java 并发编程系列: CountDownLatch (上厕所的案例)
- 详解ANGULAR2组件中的变化检测机制(对比ANGULAR1的脏检测)
- 如何快速查看github代码库中第一次commit的记录
- 更换包管理工具npm为yarn
- 【精选】使用Cryptory分析影响加密货币价格的因素(区块链系列3)
- React -- 组件间通信
- 图片和视频防盗链简单介绍
- 对比cp和scp命令 将数据从一台linux服务器复制到另一台linux服务器
- laravel—用Migration的操作数据库
- 有货移动Web端性能优化探索实践
- webpack打包速度和性能再次优化
- MySQL 清除表空间碎片
- 解决ios不支持按钮:active伪类的方法
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Android第三方登录之QQ登录
- Android利用CountDownTimer实现验证码倒计时效果实例
- Android开发实现读取Assets下文件及文件写入存储卡的方法
- Android studio实现刮刮乐的方法
- Android studio圆形进度条 百分数跟随变化
- Android中的SpannableString与SpannableStringBuilder详解
- 浅谈Android 中图片的三级缓存策略
- Android主线程和子线程区别详解
- Android 用RxBinding与RxJava2实现短信验证码倒计时功能
- Android右滑返回上一个界面的实现方法
- Android pull解析xml的实现方法
- Android实现QQ图片说说照片选择效果
- 一个简单的toolabar结合drawlayout使用方法
- 利用DrawerLayout和触摸事件分发实现抽屉侧滑效果
- Android App端与PHP Web端的简单数据交互实现示例