博客地址:http://www.cnblogs.com/yudanqu/

一、递归

  • 递归调用:一个函数,调用的自身,称为递归调用
  • 递归函数:一个可以调用自身的函数称为递归函数

  凡是循环能干的事,递归都能干

方法:
1、写出临界条件
2、找这一次和上一次的关系
3、假设当前函数已经能用,调用自身计算上一次的结果再求出本次的结果

  下面我们通过两段代码简单看一下递归和非递归的区别:

    输入一个大于等于1的数,求1到n的和!

 # 普通函数方法

 def hanshu(n):
sum = 0
# 循环遍历每一个数字,将他们加到一个事先定义好的变量上,直到加完
for x in range(1, n+1):
sum += x
return sum

  下面看一下通过递归的方法:

 # 递归

 def digui(n):
if n == 1:
return 1 # 如果n等于1证明已经递归到最后,返回1,这就是上述的临界条件
else:
return n + digui(n-1) # 当没有达到临界条件时,用n加上对n-1的递归,每次都把n加进去,但是后面依然是使用当下这个递归函数,会再次调用计算n-1,直到递归结束,也就是将从n到1的数全部递归完

  在实际应用中,递归是十分消耗内存的,但是有些事情他很容易去做,很容易理解。下面,就通过一个案例介绍一下递归的用法。

二、递归遍历目录

  下面的内容我就通过解释代码来讲解了,如果哪里讲的不清楚,欢迎大家下方评论提意见。

 import os # 由于我们遍历目录,所以要找到那个目录并操作,os模块包含普遍的操作系统功能

 path = "" # 这是我们要遍历的目录的路径,需要自己写进去

 # 既然是递归函数,那么肯定要有个函数,而且这个函数还将在函数内部再次被调用
def getAllDir(path, sp = ''): # 参数中传入路径和sp,这个我最后说一句你就明白了
# 得到当前目录下的所有文件
filesList = os.listdir(path) # os.listdir()是os模块下的一个方法,相当于Linux中的ls,查看所有文件 sp += " " # 这个也先放一下
# 处理每一个文件
for fileName in filesList: # 遍历刚才找到的目录下的所有文件
# 判断是否是目录(要用绝对路径)
fileAbsPath = os.path.join(path,fileName) # join是os模块下将两个路径拼接在一起的意思,第二个参数不能有斜杠。因为我们要判断一下这个文件是一个普通文件还是一个目录,所有要先把他的绝对路径整理出来
if os.path.isdir(fileAbsPath): # isdir是判断是否为目录,是则返回True
print(sp + "目录:", fileName) # 打印当前这个文件,他是个目录
getAllDir(fileAbsPath,sp = " ") # 这里就开始递归了,因为我们要找到整个目录里的东西,所以当这个文件还是个目录的时候我们需要继续向下找
else:
print(sp + "普通文件:", fileName) # 如果仅仅是个普通文件,那么他里面也就没有其他文件了,就可以直接打印他了 getAllDir(path) # 这里是调用函数,让遍历开始 # 最后我来说一下开始写的那个sp,是space的意思,有人也许现在就明白了。那个其实就是让我们方便观察,因为每次打印都是顶行写的,我们分不清他的目录结构,所以通过空格来调整。在函数内部写一个空格增加的表达式,可以使调用次数和空格数相关起来,递归的越深,证明目录的级越低,那么空格越多

三、栈模拟递归遍历目录(深度遍历)

 # 整体思路是没有变得,这里没有写到的也许是重复的,看下上面注释就好了
# 写了一半想起来应该回来写一下栈:栈就是一个容器,但它只有一个口,入栈出栈都从这一个口,而且这个栈很细,进去了就不能颠倒位置了,所以,每入栈一个元素他在最外面时候可以出来,否则得等前面的走完了它才可以出来
import os def getAllDirDFS(path):
stack = [] # 这里是用栈来模拟,我们先创建一个列表当做栈
stack.append(path) # 首先,先向栈里压入路径 # 处理栈,当栈为空时结束循环(栈为空就说明栈里没有普通文件和目录了,因为我们是每操作一个要把那个取出来)
while len(stack) != 0:
# 从栈中取出数据
dirPath = stack.pop() # pop函数是删除最后一个元素,同时还有一个返回值,就是去除的那个元素,我们再接收一下等等用
# 目录下所有文件
filesList = os.listdir(dirPath) # 这个和上面一样 # 处理每一个文件,如果是普通文件则打印出来,如果是目录则将该目录地址压栈
for fileName in filesList:
# print(dirPath)
fileAbsPath = os.path.join(dirPath,fileName)
# print(fileAbsPath)
if os.path.isdir(fileAbsPath):
# 是目录就压栈
print("目录:" + fileName)
stack.append(fileAbsPath) # 当是目录时入栈,它这时就在最外面,下一次循环时候要取出栈的元素是不是还是这个啊,既然是它的话就还有找他内部的东西,等把他找完了才继续找和他并列的那些文件。就是说抓住一根绳子使劲往下找,找到头没有了才返回来,这就是深度优先遍历
else:
# 打印普通文件
print("普通:" + fileName) getAllDirDFS(path)

四、队列模拟递归遍历目录(广度遍历)

 # 这回记住了,先说一下队列,队是一个两端开口的模型,一头进一头出,当然还有双向队列,循环等等,我们就简单用一下最基本的队列

 import collections # 队列在python的包里有,所以我们直接调用一下,不用以为这个很难,他也只不过是类型是queue,实际的思想是一样的,入队append,因为这个是在右侧,也就是后方入队,另一边出的话就是popleft,左侧出,是不是很通俗,只是改了一下出来的口
def getAllDirBFS(path):
queue = collections.deque() # 创建一个队列,只要记得后面用法就是上面我说的那个不同就可以了
queue.append(path) while len(queue) != 0:
dirPath = queue.popleft() # 仅仅这里不同,因为队列模拟是另一端出队
filesList = os.listdir(dirPath)
for fileName in filesList:
fileAbsPath = os.path.join(dirPath,fileName)
if os.path.isdir(fileAbsPath):
print('目录:' + fileName)
queue.append(fileAbsPath)
else:
print('文件:' + fileName) getAllDirBFS(path) # 大家想一下,栈是哪里进哪里出,也就是,刚进去的元素,接下来的一次循环又出来了,那便是一条路走到头,是深度遍历;那么现在一头进另一头出是什么意思呢,就是即便判断了这个是一个目录,但我现在不执行你,我要把你前面这些都查一遍,找完是目录的都添加在后面,之后再遍历你们这些,就是把一层的内容找完再找下一层,被称为广度优先遍历。

  写这篇随笔的时候没有考虑到也许有些同学还需要学习一下栈和队列,我会在之后再进行一下总结。本人也是初学者,希望和大家一起交流。

  作者:渔单渠(yudanqu)

  博客地址:http://www.cnblogs.com/yudanqu/

python--递归(附利用栈和队列模拟递归)的更多相关文章

  1. python之单例模式、栈、队列和有序字典

    一.单例模式 import time import threading class Singleton(object): lock = threading.RLock() # 定义一把锁 __inst ...

  2. Python将列表作为栈和队列

    Collections中的各种方法 阅读目录(Content) 一.各种方法介绍 二.代码部分 回到顶部(go to top) 一.各种方法介绍 Counter 统计个数   elements  mo ...

  3. python栈、队列、文件目录遍历

    一. 栈与队列 关注公众号"轻松学编程"了解更多. 1. 栈 stack 特点:先进先出[可以抽象成竹筒中的豆子,先进去的后出来] 后来者居上 mystack = [] #压栈[向 ...

  4. 【LeetCode题解】232_用栈实现队列(Implement-Queue-using-Stacks)

    目录 描述 解法一:在一个栈中维持所有元素的出队顺序 思路 入队(push) 出队(pop) 查看队首(peek) 是否为空(empty) Java 实现 Python 实现 解法二:一个栈入,一个栈 ...

  5. 《剑指offer》递推与循环,栈和队列,回溯 (牛客10.27)

    难度 题目 知识点 07. 斐波那契数列 递推递归 - 两变量写法- 08. 跳台阶 同上 09. 变态跳台阶 dp 10. 矩形覆盖 同上 05. 用两个栈实现队列 模拟 ☆ 20. 包含min函数 ...

  6. 【DataStructure In Python】Python模拟栈和队列

    用Python模拟栈和队列主要是利用List,当然也可以使用collection的deque.以下内容为栈: #! /usr/bin/env python # DataStructure Stack ...

  7. 两队列模拟一个栈,python实现

    python实现两个队列模拟一个栈: class Queue(object): def __init__(self): self.stack1=[] self.stack2=[] def enqueu ...

  8. python 递归深度优先搜索与广度优先搜索算法模拟实现

    一.递归原理小案例分析 (1)# 概述 递归:即一个函数调用了自身,即实现了递归 凡是循环能做到的事,递归一般都能做到! (2)# 写递归的过程 1.写出临界条件2.找出这一次和上一次关系3.假设当前 ...

  9. python数据结构之栈与队列

    python数据结构之栈与队列 用list实现堆栈stack 堆栈:后进先出 如何进?用append 如何出?用pop() >>> >>> stack = [3, ...

随机推荐

  1. ChakraCore/JSRT使用问题汇总

    QQ交流群:523723780(ChakraCore) ChakraCore是什么? 一个微软开源的,用于Windows IE/Edge内核的高效JS脚本引擎. 前不久微软开源了ChakraCore, ...

  2. 基于OpenSSL的HTTPS通信C++实现

      HTTPS是以安全为目标的HTTP通道,简单讲是HTTP的安全版.即HTTP下加入SSL层,HTTPS的安全基础是SSL,因此加密的详细内容就需要SSL.Nebula是一个为开发者提供一个快速开发 ...

  3. c/c++ 标准库 插入迭代器 详解

    标准库 插入迭代器 详解 插入迭代器作用:copy等函数不能改变容器的大小,所以有时copy先容器是个空的容器,如果不使用插入迭代器,是无法使用copy等函数的. 例如下面的代码就是错误的: list ...

  4. 特别篇:Hyper-v群集模拟实战演示

    介绍 由于前面几张的都是直接整理了下 九叔的hyper-v电子书发上来的,个人觉得他写的不是最详细,因此今天我按照自己的实际情况来写个模拟的实战演示.所有的东西都通过VMware WorkStatio ...

  5. C++实现程序单实例运行的两种方式

    简介 在我们编写程序的时候,经常会注意到的一个问题就是如何能够让程序只运行一个实例,确保不会让同一个程序多次运行,从而产生诸多相同进程,给我们的带来不便呢?那么常用的有以下四种方法,第一种方法是通过扫 ...

  6. March 04th, 2018 Week 10th Sunday

    Tomorrow never comes. 我生待明日,万事成蹉跎. Most of my past failures can be chalked up to the bad habit of pr ...

  7. jdk旧版本下载

    如何找到旧版本的jdk: 1.去oracle官网关于下载jdk的这一板块,https://www.oracle.com/technetwork/java/javase/downloads/index. ...

  8. 网络协议与OSI体系结构

    网络协议与网络体系结构 一.网络协议的概念 1.含义: 网络协议是计算机间进行通信时遵循的一些约定和规则 2.三要素: (1)语法:用于确定协议元素的格式,即数据与控制信息的结构 (2)语义:用于确定 ...

  9. E - Intervals 贪心

    Chiaki has n intervals and the i-th of them is [li, ri]. She wants to delete some intervals so that ...

  10. 【NOI2019模拟】搬砖

    [NOI2019模拟]搬砖 Description 小火车很穷,只好去搬砖了. 小火车被工头要求搭建\(n\)座塔,第i个高度为\(H_i\),也就是由\(H_i\)块砖头组成.每次小火车可以携带至多 ...