暴力递归——汉诺塔问题

简介: 暴力递归——汉诺塔问题

打印n层汉诺塔从最左边移动到最右边的全部过程

暴力递归就是尝试

1)把问题转化为规模缩小了的同类问题的子问题

2)有明确的不需要继续进行递归的条件(base case)

3)有当得到了子问题的结果之后的决策过程

4)不记录每一个子问题的解

补充一点,如果记录每一个子问题的解就是动态规划,后续文章会讲到。

递归方法

假设有3跟柱子,左、中、右,我们细分为如下三个步骤:

1)将1~N-1层圆盘从左 -> 中(大问题,需要继续划分为小问题)

2)将第N层圆盘从左 -> 右(base case)

3)将1~N-1层圆盘从中 -> 右(大问题,需要继续划分为小问题)

递归方法改进版

如果我们忘记柱子的左中右,将三根柱子标记为from、to、other。我们实现的是,将所有的圆盘(一共N层)从from -> to,同样细分为3个步骤:

1)将1~N-1层圆盘从from -> other(大问题,需要继续划分为小问题)

2)将第N层圆盘从from -> to(base case,可以直接打印)

3)将1~N-1层圆盘从other -> to(大问题,需要继续划分为小问题)

貌似没什么改进,因为还是细分这三步,同样要递归,但关键在于忘掉柱子的左中右顺序,只需要三个变量,代码可以少很多。

总结

汉诺塔问题还有非递归的解法,因为任何递归都可以改成非递归,只需要自己设计压栈。但是博主个人认为汉诺塔的递归方法比非递归更好容易理解和实现,非递归自己去设计压栈还比较麻烦,博主目前也还没弄懂,所以就只附上代码了。后面如果搞懂了在补充吧,😄

附上完整代码

package com.harrison.class12;
import java.util.Stack;
public class Code01_Hanoi {
  public static void leftToRight(int n) {
    if(n==1) {
      System.out.println("move 1 from left to right");
      return ;
    }
    leftToMid(n-1);
    System.out.println("move "+ n +" from left to right");
    midToRight(n-1);
  }
  public static void leftToMid(int n) {
    if(n==1) {
      System.out.println("move 1 from left to mid");
      return ;
    }
    leftToRight(n-1);
    System.out.println("move "+ n +" from left to mid");
    rightToMid(n-1);
  }
  public static void midToRight(int n) {
    if(n==1) {
      System.out.println("move 1 from mid to right");
      return ;
    }
    midToLeft(n-1);
    System.out.println("move "+ n +" from mid to right");
    leftToRight(n-1);
  }
  public static void midToLeft(int n) {
    if(n==1) {
      System.out.println("move 1 from mid to left");
      return ;
    }
    midToRight(n-1);
    System.out.println("move "+ n +" from mid to left");
    rightToLeft(n-1);
  }
  public static void rightToLeft(int n) {
    if(n==1) {
      System.out.println("move 1 from right to left");
      return ;
    }
    rightToMid(n-1);
    System.out.println("move "+ n +" from right to left");
    midToLeft(n);
  }
  public static void rightToMid(int n) {
    if(n==1) {
      System.out.println("move 1 from right to mid");
      return ;
    }
    rightToLeft(n-1);
    System.out.println("move "+ n +" from right to mid");
    leftToMid(n-1);
  }
  public static void hanoi1(int n) {
    leftToRight(n);
  }
  public static void f(int n,String from,String to,String other) {
    if(n==1) {
      System.out.println("move 1 from "+ from + " to " + to);
    }else {
      f(n-1, from, other, to);
      System.out.println("move " + n + " from " +from +" to "+to);
      f(n-1, other,to,from);
    }
  }
  public static void hanoi2(int n) {
    if(n>0) {
      f(n, "left", "right", "mid");
    }
  }
  public static class Record {
    public boolean finish1;
    public int base;
    public String from;
    public String to;
    public String other;
    public Record(boolean f1, int b, String f, String t, String o) {
      finish1 = false;
      base = b;
      from = f;
      to = t;
      other = o;
    }
  }
  public static void hanoi3(int N) {
    if (N < 1) {
      return;
    }
    Stack<Record> stack = new Stack<>();
    stack.add(new Record(false, N, "left", "right", "mid"));
    while (!stack.isEmpty()) {
      Record cur = stack.pop();
      if (cur.base == 1) {
        System.out.println("Move 1 from " + cur.from + " to " + cur.to);
        if (!stack.isEmpty()) {
          stack.peek().finish1 = true;
        }
      } else {
        if (!cur.finish1) {
          stack.push(cur);
          stack.push(new Record(false, cur.base - 1, cur.from, cur.other, cur.to));
        } else {
          System.out.println("Move " + cur.base + " from " + cur.from + " to " + cur.to);
          stack.push(new Record(false, cur.base - 1, cur.other, cur.to, cur.from));
        }
      }
    }
  }
  public static void main(String[] args) {
    int n=3;
    hanoi1(n);
    System.out.println("=======================");
    hanoi2(n);
    System.out.println("=======================");
    hanoi3(n);
    System.out.println("=======================");
  }
}


相关文章
|
Go Android开发 开发者
关于Xposed和Magisk的各方面比较(附Xposed框架各版本卡刷包及安装器)
说到搞机神器,不得不提江湖老大哥Xposed和后起之秀Magisk这两个框架,文章简单的介绍一下两者相似和不同之处
2693 0
|
人工智能 API 决策智能
MetaGPT( The Multi-Agent Framework):颠覆AI开发的革命性多智能体元编程框架
MetaGPT( The Multi-Agent Framework):颠覆AI开发的革命性多智能体元编程框架
MetaGPT( The Multi-Agent Framework):颠覆AI开发的革命性多智能体元编程框架
|
6月前
|
机器学习/深度学习 算法 索引
YOLOv11改进 - 卷积Conv | 加权卷积wConv:无损替换标准卷积,增强空间建模与特征提取质量
本文提出加权卷积(wConv),通过引入距离感知的密度函数,自适应调整邻域像素权重,突破传统卷积等权局限。结合双优化器策略,在不增加参数量的前提下提升特征提取能力。集成于YOLOv11后显著降低损失、提高检测精度,适用于图像去噪等密集预测任务。
|
人工智能 自然语言处理 测试技术
通义灵码上新推理模型,快来体验数学编程双冠王 Qwen2.5-Max
近日,通义灵码上新模型选择功能,除新增 DeepSeek 满血版 V3 和 R1 外,Qwen2.5-Max 也正式上线,它使用了超过 20 万亿 token 的预训练数据及精心设计的后训练方案进行训练。
|
算法 开发者 索引
【C++11算法】random_shuffle和shuffle
【C++11算法】random_shuffle和shuffle
1134 0
|
缓存 网络协议 网络性能优化
网络协议详解:TCP/IP与HTTP
【7月更文挑战第24天】TCP/IP协议和HTTP协议是现代互联网通信的重要基石。TCP/IP协议提供了计算机之间数据传输和通信的底层支持,而HTTP协议则在此基础上实现了超文本数据的传输。随着互联网的不断发展,TCP/IP协议和HTTP协议将继续发挥重要作用,为各种网络应用提供稳定、高效的通信服务。
1007 3
STM32CubeMX 按键控制LED
STM32CubeMX 按键控制LED
602 0
|
JavaScript 容器
模态框(Modal
模态框(Modal)是一种用于在网页上展示重要信息或功能的交互式窗口。它通常在页面顶部或页面中部弹出,覆盖在页面之上,使页面部分内容不可见,直到模态框被关闭。模态框可以包含文本、图像、表单、按钮等元素,用于向用户展示信息、获取用户输入或执行其他操作。
794 4
|
数据采集 大数据 数据管理
Veracity(真实性)
Veracity(真实性)
1176 2
|
SQL 关系型数据库 MySQL
数据库字段基本类型
数据库字段基本类型
554 0