关于递归和回溯的一次深入思考

业余算法coder,平时做得最多的数据结构算法就是模拟,很久之前学过递归,后来接触到回溯之后,一直很懵,同样的递归,回溯除了要进行“复原”以外,为什么会多一个for循环。之前一直没搞懂这个问题,也没有去深究。直到昨天lc的每日一题,我一眼看出来可以用递归解,用递归写了半天都不会,然后看大佬写的回溯,又是for循环中去递归,就好像是以前的质变引起量变了一样,我突然就悟了。

贴一道经典递归题:打家劫舍

“有一个数组values=[5,9,6,2,4,1,3,7,10],小偷不能偷相邻的财报,也就是说,小偷偷了第一个5之后,就不能偷第二个9了”

这种就是标准的递归二叉树结构:

打勾的表示拿,打叉的表示不拿,如果拿了5,就只有考虑6拿不拿了,如果不拿5,就可以考虑9拿不拿,就这样一直决策下去,取价值最大的一种方法。

代码如下:

        public int dfs(int[] nums, int idx, int curValue) {
            if (idx >= nums.length) {
                return curValue;
            }
	    // 当前这个拿,去下一个决策
	    int no = dfs(nums, idx + 1, curValue);
			
	    // 当前这个拿,下一个就肯定不能拿了,得去下一个的下一个做决策
            int yes = dfs(nums, idx + 2, curValue + nums[idx]);

            return Math.max(no, yes);
        }

再来贴一道经典回溯题:打印子序列

假如有一个字符串 abcd,那么它的子序列为 : a,b,c,d,ab,ac,ad,bc,bd,cd

这种就是标准的回溯结构:

也就是在这里,我搞明白了为什么需要一个for'循环,上面的模型,只有一棵树,而回溯,每个点都能成为一棵树,所以每个for循环的时候,就是以这个点开始遍历树。

  private void dfs(char[] str, int index, HashSet<String> ans, String path) {
    if (index >= str.length) {
      ans.add(path);
      return;
    }
    // 分别以 a,b,c,d 为头节点遍历整棵树
    for (int j = index; j < str.length; j++) {
      // 当前值拿的情况,去遍历
      path += str[j];
      dfs(str, j + 1, ans, path);
      // 回溯精髓:复原
      path = path.substring(0, path.length() - 1);
      // 当前值不拿的情况,去遍历
      dfs(str, j + 1, ans, path);
    }
  }

后面会更新一下从递归到记忆化搜索和动态规划的方法,动态规划并不是一蹴而就,转移方程也不是直接看出来的,原始的方法就是从递归优化到动态规划。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
递归回溯和迭代回溯都是解决问题的常见方法,它们的主要区别在于求解问题的方式和实现方式。 1. 递归回溯递归回溯是通过递归调用函数本身来解决问题的方法。在每一层递归中,它会尝试所有可能的解决方案,并在找到解决方案或无法继续时进行回溯。具体步骤如下: - 确定递归函数的参数和返回值。 - 判断递归终止条件,即找到解决方案或无法继续。 - 在每一层递归中,尝试所有可能的解决方案。 - 如果找到解决方案,返回结果;否则,进行回溯,撤销上一步操作,继续尝试其他可能的解决方案。 递归回溯的优点是代码简洁,易于理解和实现。但是,在某些情况下,递归会导致函数调用栈溢出,并且可能存在重复计算的问题。 2. 迭代回溯: 迭代回溯是通过使用循环和栈来模拟回溯过程的方法。它不使用函数的递归调用,而是手动管理状态和回溯的过程。具体步骤如下: - 使用栈保存当前状态。 - 判断循环终止条件,即找到解决方案或无法继续。 - 在每一次循环中,尝试所有可能的解决方案。 - 如果找到解决方案,返回结果;否则,进行回溯,撤销上一步操作,继续尝试其他可能的解决方案。 迭代回溯的优点是避免了函数调用栈溢出的问题,并且可以更灵活地控制回溯的过程。但是,相比于递归回溯,迭代回溯的实现会稍微复杂一些。 综上所述,递归回溯和迭代回溯都是解决问题的有效方法,选择哪种方法取决于具体的问题和个人偏好。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值