递归、搜索与回溯算法 2024.7.4-24.7.9

专题介绍:

image.png
image.pngimage.png

一、递归

1、汉诺塔问题

image.png

class Solution {public void hanota(List<Integer> A, List<Integer> B, List<Integer> C) {int n = A.size();move(n,A,B,C);// 将A柱上的n个盘子通过借助B盘子全部挪到C柱子上}void move(int m,List<Integer> A, List<Integer> B, List<Integer> C) {if (m == 1) {C.add(A.remove(A.size()-1));} else {// 将A柱上的m-1个盘子通过借助C盘子全部挪到B柱子上,此时已经对A数列进行了元素删除move(m-1,A, C, B); // 挪动A柱子上的第1个盘子,也就是在没有对A进行修改时的第m个盘子。// 这里容易写错成 C.add(A.remove(0));C.add(A.remove(A.size()-1));// 将B柱上的m-1个盘子通过借助A盘子全部挪到C柱子上,此时已经对A数列进行了元素删除move(m-1,B, A, C); }}
}
// 蛮细节的一道题目
class Solution {public void hanota(List<Integer> A, List<Integer> B, List<Integer> C) {int n=A.size();move(n,A,B,C);}void move(int n, List<Integer> A, List<Integer> B, List<Integer> C) {if (n == 1) {C.add(A.remove(A.size() - 1));} else {// 这样写会有大盘子压小盘子的问题B.add(A.remove(A.size() - 1));move(n - 1, A, B, C); C.add(B.remove(B.size() - 1));}}
}
// 思路错误,大盘子不能压小盘子

递归的思想,将问题转化为规模更小的子问题

// 相信move函数可以解决子问题
move(m-1,A, C, B);  
C.add(A.remove(A.size()-1));
move(m-1,B, A, C); 
// 通过自己调用自己解决原问题

image.png

2、合并两个有序链表

image.png

class Solution {public ListNode mergeTwoLists(ListNode list1, ListNode list2) {if(list1==null) return list2;if(list2==null) return list1;ListNode ret = new ListNode();if(list1.val<list2.val){ret.next = list1;list1 = list1.next;ListNode temp = mergeTwoLists(list1,list2);ret.next.next = temp;}else{ret.next = list2;list2 = list2.next;ListNode temp = mergeTwoLists(list1,list2);ret.next.next = temp;}return ret.next;}
}class Solution {public ListNode mergeTwoLists(ListNode list1, ListNode list2) {if(list1==null) return list2;if(list2==null) return list1;ListNode ret = new ListNode();if(list1.val<list2.val){ret = list1;list1 = list1.next;ListNode temp = mergeTwoLists(list1,list2);ret.next=temp; // 这一步很重要}else{ret = list2;list2 = list2.next;ListNode temp = mergeTwoLists(list1,list2);ret.next=temp; // 这一步很重要}return ret;}
}

尽量写一个递归函数就完成问题,汉诺塔问题递归函数需要四个参数,而题目只给了三个参数,所以需要创建一个单独的递归函数,然后使用驱动函数调用这个递归函数。

image.png

void main(int[] nums){dfs(arr,0);
}
void dfs(int[] arr,int i){// 顺序打印cout << arr[i]<<' ';dfs(arr,i+1);// 逆序打印,只需要调换一下打印和调用的顺序即可。 前序遍历VS逆序遍历// dfs(arr,i+1);// cout << arr[i]<<' ';
}

3、反转链表

image.png
image.png
image.png

class Solution {public ListNode reverseList(ListNode head) {if (head == null || head.next == null)return head;ListNode temp = reverseList(head.next);head.next.next = head;head.next=null;        return temp;}
}

image.png

递归函数:将以head为头结点的链表逆序,然会一个新的指针指向逆序链表的头结点,head依然指向原来的结点也就是逆序列表的为节点。
递归函数应该还可以这样去定义:将以head为头结点的链表针逆序,逆序后head指向逆序列表的头结点,该递归函数无需返回值,有也行。应该可以,挖坑。

4、两两交换链表中的结点

image.png

class Solution {public ListNode swapPairs(ListNode head) {if(head==null||head.next==null) return head;ListNode next = head.next;head.next = head.next.next;next.next = head;ListNode temp = swapPairs(head.next);head.next = temp;return next;}
}

将递归函数看成一个黑盒子,相信他可以解决子问题。

5、Pow(x, n)

image.png

class Solution {public double myPow(double x, int n) {if(n>=0) return fun(x,n);else return 1/fun(x,-n);}public double fun(double x,int n){if(n==0) return 1;double a = fun(x,n/2); // 两次用到这个值,提前计算出来减少递归次数if(n%2==0){return a*a;}else{return a*a*x;}}
}
// 可以通过
class Solution {public double myPow(double x, int n) {if(n>=0) return fun(x,n);else return 1/fun(x,-n);}public double fun(double x,int n){if(n==0) return 1;double a = fun(x,n/2); // 两次用到这个值,提前计算出来减少递归次数if(n%2==1){return a*a*x;}else{return a*a;}}
}
// 不能通过
// 当n=-n=-2147483648,-n也等于n=-2147483648
// 在不断调用fun的过程中n始终小于等于0,n%2不可能等于1// fun(x,-2147483648)最终结果等于1,导致结果给错误
// fun(double x,int x);必须保证n>=0;对于这种特例要么特判,要么就是使用long整形进行传参
class Solution {public double myPow(double x, int n) {long m = (long)n;if(m>=0) return fun(x,m);else return 1/fun(x,-m);}public double fun(double x,long n){if(n==0) return 1;double a = fun(x,n/2); // 两次用到这个值,提前计算出来减少递归次数if(n%2==1){return a*a*x;}else{return a*a;}}
}

二、二叉树中的深搜

6、计算布尔二叉树的值

image.png

class Solution {public boolean evaluateTree(TreeNode root) {if(root.val==0) return false;if(root.val==1) return true;boolean left = evaluateTree(root.left);boolean right = evaluateTree(root.right);// 我是瞎子,我把这里写成了root.rightif(root.val==2) return left || right;return left && right;}
}
// 整个代码就是后续遍历

原问题:根据题意计算root的值
子问题:根据题意计算某个分支节点的值
最小子问题:叶子节点
因为问题中只牵扯了root,所以题目给的函数就是递归函数

7、求根节点到叶子节点数字之和 ★★★★

image.png
image.png

class Solution {public int sumNumbers(TreeNode root) {return sum(0,root);}public int sum(int val,TreeNode root){// 到达叶子节点就返回这条路径的数值if(root!=null&&root.left==null&&root.right==null) return val*10+root.val;int sum = 0;// 有叶子节点就继续递归if(root.left!=null) sum += sum(val*10+root.val,root.left);if(root.right!=null) sum += sum(val*10+root.val,root.right);return sum;}
}

遍历到叶子节点的时候路径的数组已经被算出来了,然后返回相加即可。
子问题:有一个正数val和一个二叉树root,求从根节点到叶子节点构成的所有数字之和
原问题:val=0,二叉树为root。
子问题:val为根节点到分支节点构成的数字,分支节点为根节点的二叉树
问题中牵扯两个变量,递归函数有两个参数,因此需要自定义一个递归函数,使用题目给定的函数作为驱动函数。

8、二叉树减枝头 ★★★★(函数函数传参)

image.png

class Solution {public TreeNode pruneTree(TreeNode root) {// 原问题:对root进行剪枝// 子问题:对某个分支节点为根节点的二叉树进行剪枝// 最小子问题(递归出口):叶子节点// 0、递归出口,val为0的叶子节点if (root == null)return null;// 1、修剪左右子树TreeNode left = pruneTree(root.left);TreeNode right = pruneTree(root.right);// 2、修剪之后重新判断当前结点是否需要修剪if (left == null && right == null && root.val == 0)// 这一层的root是上一层的left或者right,修改root难道对上一层没有修改吗?root = null;// 3、返回值return root;// 这个root只是临时变量吗???}
}
// 调试代码会发现,pruneTree并未对root做任何修改,说我只是修改了修改了临时变量
// 什么时候剪枝?
class Solution {public TreeNode pruneTree(TreeNode root) {// 原问题:对root进行剪枝// 子问题:对某个分支节点为根节点的二叉树进行剪枝// 最小子问题(递归出口):叶子节点// 0、递归出口,val为0的叶子节点if (root == null)return null;// 1、修剪左右子树root.left = pruneTree(root.left);// 此处发生剪枝root.right = pruneTree(root.right);// 此处发生剪枝// 2、修剪之后重新判断当前结点是否需要修剪if (root.left == null && root.right == null && root.val == 0)root = null;// 3、返回值return root;}
}

错误题解的允许过程
正确题解的运行过程
在Java中,所有函数参数传递都是“按值传递”的。但这里需要注意的是,对于基本数据类型(如int、char等)和对象类型,"按值传递"的行为是不同的:

  1. 基本数据类型
public static void modifyValue(int x) {x = 10;
}public static void main(String[] args) {int a = 5;modifyValue(a);System.out.println(a); // 输出仍为5
}
  • 对于基本数据类型(如int、char等),Java会复制变量的值并传递给函数。因此,函数内对参数的修改不会影响函数外的原始变量。
  1. 对象类型
class MyObject {int value;
}public static void modifyObject(MyObject obj) {obj.value = 10; // 修改对象的属性,影响到原始对象
}public static void reassignObject(MyObject obj) {obj = new MyObject(); // 修改对象引用,不影响原始对象obj.value = 20;
}public static void main(String[] args) {MyObject obj = new MyObject();obj.value = 5;modifyObject(obj);System.out.println(obj.value); // 输出10reassignObject(obj);System.out.println(obj.value); // 输出10,reassignObject中的修改无效
}
  • 对于对象类型(如类的实例),Java会复制对象引用的值并传递给函数。函数内部对对象属性的修改会影响到原始对象,因为它们共享同一个对象引用。
  • 但如果在函数内部改变对象引用本身,这种改变不会影响到函数外的引用。

**pruneTree**函数中的表现:

public static TreeNode pruneTree(TreeNode root) {if (root == null)return null;TreeNode left = pruneTree(root.left);TreeNode right = pruneTree(root.right);if (left == null && right == null && root.val == 0)root = null;return root;
}

在这个函数中,root是一个对象引用。当函数被调用时,root的引用被复制传递给函数。因此,对root引用的修改(如将root设为null)并不会影响到原始的树结构,除非我们显式地将修改后的左子树和右子树重新赋值给root的左子树和右子树引用。
因此,修改后的版本如下:

public static TreeNode pruneTree(TreeNode root) {if (root == null)return null;root.left = pruneTree(root.left); // 正确修改左子树root.right = pruneTree(root.right); // 正确修改右子树if (root.left == null && root.right == null && root.val == 0)return null; // 返回null剪掉当前节点return root;
}

这样做确保了递归调用的结果被正确赋值回root的左右子树指针,从而实现对树的有效剪枝。

9、验证二叉搜索树

image.png

判断是否时二叉搜索树的方法:

  1. 根据定义来判断 (递归函数需要三个参数)
  2. 中序遍历二叉树的结果说递增的(使用一个全局变量标记前一个节点的值)

image.png

class Solution {long prve = Long.MIN_VALUE;public boolean isValidBST(TreeNode root) {if(root == null) return true;boolean left = isValidBST(root.left);boolean cur = false;if(root.val > prve) cur = true;prve = root.val;boolean right = isValidBST(root.right);return left && cur && right;}}
class Solution {long prve = Long.MIN_VALUE;// 中序遍历public boolean isValidBST(TreeNode root) {if(root == null) return true;// 判断右节点if(!isValidBST(root.left)) return false; //提前返回,剪枝操作// 判断根节点if(root.val<=prve) return false; // 提前返回,剪枝操作// 更新prevprve = (long)root.val;// 判断左节点return isValidBST(root.right);//剪枝操作}}

10、二叉搜索树中第k小的元素 ★★★★★(剪枝)

image.png

class Solution {// k<=n 所以题目保证有结果// 中序遍历二叉搜索的结果是递增的 => 寻找中序遍历的第k个数int index = 0;public int kthSmallest(TreeNode root, int k) {if (root == null)return -1;// 使用-1表示该节点不是第k小节点int left = kthSmallest(root.left, k);if (left != -1)return left;index++;if (index == k)return root.val;return kthSmallest(root.right, k);}
}
// 仔细观察可以发现,递归函数并不需要参数k
class Solution {int count;int ret;public int kthSmallest(TreeNode root, int k) {count = k;dfs(root);return ret;}// 返回值可能存在与右子树、根节点以及左子树,所以三个地方都可以进行剪枝操作// 当已经找到答案的时候,后续的遍历都可以之间返回了。void dfs(TreeNode root) {if (root == null || count == 0) //剪枝一return;dfs(root.left);if(count==0) return; //剪枝二count--;if (count == 0)ret = root.val;if (count == 0) // 剪枝三return;dfs(root.right);}
}
class Solution {int count;public int kthSmallest(TreeNode root, int k) {// 如果让count从0加到k,那么递归函数就需要k这个参数// 如果初始化count=k,让count减到0,递归函数一个就不需要k这个参数了count = k; return dfs(root);}int dfs(TreeNode root) {// count=0的作用:count是个全局变量,当count减到0的时候已经找到返回值了,后续的递归调用就不用再进行了,如果再进行count就会变成负数。if (root == null || count == 0) // 剪枝一return -1;int left = dfs(root.left);if(left!=-1) return left; // 剪枝二count--;if (count == 0) // 剪枝三return root.val;return dfs(root.right);}
}

剪枝操作是机器学习和数据挖掘中的一种经常使用的技术,它的目的是减少模型的复杂性,防止过拟合,提高模型的泛化能力。剪枝的主要原理如下:

  1. 识别模型中相对重要性较低的参数或特征。这些参数或特征对模型的预测性能贡献较小,可以被删除而不会显著降低模型的性能。
  2. 通过移除这些不重要的参数或特征,可以减少模型的复杂度,从而降低过拟合的风险。过拟合通常会导致模型在训练集上表现良好,但在新的数据上表现较差。
  3. 剪枝后的模型通常更加简单和高效,在部署和使用时会更加快速和节省资源。

常见的剪枝方法包括:

  1. 基于阈值的剪枝:删除权重绝对值小于某个阈值的参数。
  2. 基于敏感度的剪枝:删除对模型性能影响较小的参数。
  3. 基于贪心算法的剪枝:递归地删除最不重要的参数。
  4. 基于正则化的剪枝:通过L1或L2正则化鼓励参数稀疏。

总之,剪枝是一种非常有效的技术,可以帮助我们构建更加简单和高效的模型,提高模型的泛化能力。在实际应用中,需要根据具体问题选择合适的剪枝方法。

剪枝操作的核心是在不显著降低模型性能的前提下,尽可能地减少模型的复杂度。这个核心可以概括为以下几个方面:

  1. 减少过拟合风险
    • 过度复杂的模型容易发生过拟合,在训练集上表现良好但在新数据上泛化性能较差。
    • 通过剪枝减少模型复杂度,可以有效降低过拟合的风险,提高模型的泛化能力。
  2. 提高模型效率
    • 更简单的模型通常计算开销更小,在部署和应用时会更加快速和节省资源。
    • 这对于一些对实时性和资源消耗有严格要求的场景非常重要,如移动端应用、嵌入式系统等。
  3. 增强模型可解释性
    • 复杂的模型往往难以解释内部工作原理,而简单的模型通常更加可解释。
    • 可解释性有利于模型的调试和修改,也有助于用户对模型的信任和接受。
  4. 避免不必要的复杂性
    • 过于复杂的模型并非总是能提供更好的性能。相反,过度复杂可能带来不必要的开销。
    • 通过剪枝找到适度复杂度的模型,既可以保证性能,又可以提高效率和可解释性。

综上所述,剪枝操作的核心就是在保持模型性能的前提下,尽可能地简化模型结构,从而降低过拟合风险、提高模型效率和可解释性。这是模型优化的重要手段之一。

11、二叉树的所有路径 ★★★★(回溯、恢复现场)

image.png

class Solution {// 全局变量List<String> list = new ArrayList<>();public List<String> binaryTreePaths(TreeNode root) {dfs(root,"");return list;}public void dfs(TreeNode root,String path){if(root == null) return;// 叶子节点if(root != null && root.left==null && root.right==null ){path=path+"->"+root.val;list.add(path.substring(2,path.length()));return ;}path=path+"->"+root.val;dfs(root.left,path);dfs(root.right,path);}
}
// path=path+"->"+root.val;该操作会new出来一个新的String对象,
// 对驱动函数中的path没有影响,因此手动实现“回复现场”
// path=path+"->"+root.val;这样操作会提高会消耗较长时间
class Solution {// 全局变量List<String> list = new ArrayList<>();public List<String> binaryTreePaths(TreeNode root) {dfs(root,new StringBuffer(""));return list;}public void dfs(TreeNode root,StringBuffer path){if(root == null) return;// 叶子节点if(root != null && root.left==null && root.right==null ){path.append("->");path.append(root.val);list.add(path.substring(2,path.length()));return ;}path.append("->");path.append(root.val);dfs(root.left,path);dfs(root.right,path);}
}
// 被掉函数会修改StringBuffer对象的值,此时驱动函数的StringBuffer对象也会被修改,
// 如果不回溯就会影响递归调用

没有回溯导致的错误

class Solution {// 全局变量List<String> list = new ArrayList<>();public List<String> binaryTreePaths(TreeNode root) {dfs(root,new StringBuffer());return list;}public void dfs(TreeNode root,StringBuffer _path){// 回溯方法一:赋值一个临时变量StringBuffer path = new StringBuffer(_path);if(root == null) return;// 算不上剪枝,遍历到空节点了path.append(root.val);if(root.left==null && root.right==null ){list.add(path.toString());return ;}path.append("->");dfs(root.left,path);// 回溯方法二:删除"->"dfs(root.right,path);}
}
class Solution {// 全局变量List<String> list = new ArrayList<>();public List<String> binaryTreePaths(TreeNode root) {dfs(root,new StringBuffer());return list;}public void dfs(TreeNode root,StringBuffer _path){StringBuffer path = new StringBuffer(_path);path.append(root.val);if(root.left==null && root.right==null ){list.add(path.toString());return ;}path.append("->");if(root.left!=null) dfs(root.left,path);if(root.right!=null) dfs(root.right,path);}
}

三、穷举vs暴搜vs深搜vs回溯vs剪枝

12、全排列 ★★★★(经典例题)

image.png
决策树:剪枝与回溯

全排列的结果在叶子节点上

image.png

class Solution {List<List<Integer>> ret = new ArrayList<>();public List<List<Integer>> permute(int[] nums) {dfs(nums,0,nums.length);return ret;}public void dfs(int[] nums,int index,int n){if(index == n){ret.add(Arrays.stream(nums).boxed().collect(Collectors.toList()));}for(int i=index;i<n;i++){swap(nums,index,i);dfs(nums,index+1,n);swap(nums,index,i);}}void swap(int[] nums,int i,int j){int temp = nums[i];nums[i] = nums[j];nums[j] = temp;}
}
class Solution {List<List<Integer>> ret = new ArrayList<>();public List<List<Integer>> permute(int[] nums) {List<Integer> list = Arrays.stream(nums).boxed().collect(Collectors.toList());dfs(list,0,nums.length);return ret;}public void dfs(List<Integer> list,int index,int n){if(index == n){ret.add(new ArrayList<>(list));}for(int i=index;i<n;i++){swap(list,index,i);dfs(list,index+1,n);swap(list,index,i);}}void swap(List<Integer> list,int i,int j){int a = list.get(i);int b = list.get(j);list.set(i,b);list.set(j,a);}
}

决策树:剪枝与回溯

class Solution {List<List<Integer>> ret;List<Integer> path;boolean[] cheak; // 用于标记第i个是否被使用过public List<List<Integer>> permute(int[] nums) {ret = new ArrayList<>();    path = new ArrayList<>();cheak = new boolean[nums.length];dfs(nums);  return ret;  }void dfs(int[] nums){if(path.size()==nums.length){// ret.add(path); // 这样不行,需要复制新的对象ret.add(new ArrayList<>(path));return;}for(int i=0;i<nums.length;i++){if(!cheak[i]){ // 剪枝。已经用过的数字不能继续使用path.add(nums[i]);cheak[i] = true;dfs(nums); // 递归调用cheak[i] = false; // 回溯path.remove(path.size()-1); // 回溯}}}
}

13、子集

image.png
方法一的决策树,子集都在叶子节点上
方法二的决策树,一条路径就是一个子集

class Solution {//全局变量的好处就在于,不需要复杂的传参List<List<Integer>> ret;List<Integer> path;public List<List<Integer>> subsets(int[] nums) {//位置连个引用变量分配内存ret = new ArrayList<>();path = new ArrayList<>();//调用递归函数dfs(nums,0);return ret;}//算法一:从第0层开始,每一层的num[i]都面临选或者不选,最后的结果是一颗满二叉树,从根节点到达所有的接子节点的路径就是我们的答案,也就是说每一个叶子节点代表一个子集。public void dfs1(int[] nums,int index){if(index==nums.length){//从始至终就只有一个path,如果不复制path,那么在添加时是正确的,之后会被该改变,最后回溯到根节点,所有之前添加的path都同时变成了nullret.add(new ArrayList<>(path));return ;}//不选dfs(nums,index+1);//选择path.add(nums[index]);dfs(nums,index+1);//回溯path.remove(path.size()-1);}//算法二:对于任何一个节点node,我们只能选择节点代表的路径path的最后一个元素(不包含该元素)到nums的最后一个元素之间的一个元素。这种决策树的每一个节点都代表一个结果,算法二明显优于算法一。public void dfs(int[] nums,int index){//不选择,我需要把当前节点代表的子集加入到ret里面ret.add(new ArrayList<>(path));//选择for(int i=index;i<nums.length;i++){path.add(nums[i]);dfs(nums,i+1);path.remove(path.size()-1);}}
}
class Solution {List<List<Integer>> ret;List<Integer> subset;public List<List<Integer>> subsets(int[] nums) {ret = new ArrayList<>();subset = new ArrayList<>();dfs(nums,0);return ret;}void dfs(int[] nums,int index){  // ret.add(new ArrayList<>(subset)); 如果写在这里就不用对空集特殊处理了for(int i=index;i<nums.length;i++){subset.add(nums[i]);// 下面这一句写在这里好理解一些ret.add(new ArrayList<>(subset));dfs(nums,i+1);subset.remove(subset.size()-1);}}}           
//                   决策树
//                    ( ) 
//         1            2           3
//     2       3    3            
//  3//                   决策树
//                    ()
//            ()             1
//        ()     2       ()     2
//      ()  3 ()  3  ()  3  ()  3
// 一共有nums.length层

解法二更优秀一些

四、综合练习

14、找出所有子集的异或总和再求和

image.png
决策树,每条路径就是一个子集

class Solution {int ret=0;// 返回值int subset=0;//子集所有元素进行异或public int subsetXORSum(int[] nums) {dfs(nums,0);return ret;}void dfs(int[] nums,int index){ret+=subset;for(int i=index;i<nums.length;i++){subset^=nums[i];dfs(nums,i+1);//这里是i+1,不是index+1;subset^=nums[i];}}
}

根据题目意思需要求集合的所有子集,采用上题的方法二

15、全排列II ★★★(去重 —>排序去重法)

image.png
image.png
image.png

class Solution {List<List<Integer>> ret;List<Integer> path;boolean[] cheak;public List<List<Integer>> permuteUnique(int[] nums) {ret = new ArrayList<>();path = new ArrayList<>();cheak = new boolean[nums.length];Arrays.sort(nums);dfs(nums);return ret;}public void dfs(int[] nums){if(path.size()==nums.length){ret.add(new ArrayList<>(path));return;}for(int i=0;i<nums.length;i++){// 关注不合法的情况// 剪枝if(cheak[i]==true|| (i!=0&&nums[i]==nums[i-1]&&cheak[i-1]==false)){continue;   }// 关注合法的情况 ture改成false,&&改成||,||改成&& 全部相反即可// if(cheak[i]==false && (i==0||nums[i]!=nums[i-1] || cheak[i-1]==true{ path.add(nums[i]);cheak[i]=true;dfs(nums);path.remove(path.size()-1);cheak[i]=false;// }}}
}

如何让去重:可以先对数组进行排序

16、电话号码的字母组合数

image.png
image.png

class Solution {String[] str={"","","abc","def","ghi","jkl","mno","pqrs","tuv","wxyz"};List<String> ret;StringBuffer path;public List<String> letterCombinations(String digits){ret = new ArrayList<>();path = new StringBuffer();if(digits.equals("")) return ret;dfs(digits,0);return ret;}void dfs(String digits,int index){// 根据题意,当解析出来的字符串长度等于if(path.length() == digits.length()){ret.add(path.toString());return ;}// 根据号码取字符串int i = digits.charAt(index)-'0';String s = str[i];// dfsfor(char ch:s.toCharArray()){path.append(ch);dfs(digits,index+1);path.deleteCharAt(path.length()-1);}}
}

digits.lengt()=N
其实就是N重循环,但是不可能真的写的N层for循环而是写递归调用。

17、括号生成

image.png
image.png

class Solution {List<String> ret;StringBuffer path;int left = 0;int right = 0;int length;public List<String> generateParenthesis(int n) {length = 2*n;ret = new ArrayList<>();path = new StringBuffer();dfs(n);return ret;}void dfs(int n){if(path.length()==length){ret.add(path.toString());return ;}// 加左边括号if(n>0){path.append('(');left++;dfs(n-1);left--;path.deleteCharAt(path.length()-1);//回溯}// 当右括号多与左括号时才能加左括号if(left>right){path.append(')');right++;dfs(n);right--;path.deleteCharAt(path.length()-1);//回溯}}
}

决策树:如果n>0,则可以加左括号;如果left>right,则可以加右括号。
当path的长度等于2*n时,说明找到了一种生成方式,加入到ret中

18、组合

image.png
决策树

class Solution {List<List<Integer>> ret;List<Integer> subset;public List<List<Integer>> combine(int n, int k) {ret = new ArrayList<>();subset = new ArrayList<>();dfs(1,n,k);return ret;}// start:从index位置开始遍历// n:[1~n]的组合// k:组合长度为kvoid dfs(int start,int n,int k){if(subset.size() == k){ret.add(new ArrayList<>(subset));return ;}// 只枚举更大的数子for(int i=start;i<=n;i++){subset.add(i);dfs(i+1,n,k);subset.remove(subset.size()-1);}}
}

19、目标和

image.png
image.png

class Solution {int ret;int target;int sum;public int findTargetSumWays(int[] nums, int _target) {target = _target;dfs(nums,0);return ret;}void dfs(int[] nums,int index){if(index == nums.length){if(sum==target) ret++;return ;}// +sum+=nums[index];dfs(nums,index+1);sum-=nums[index];// - sum-=nums[index];dfs(nums,index+1);sum+=nums[index];}
}
// 此时对于全局变量必须要搞对称操作进行回复现场
class Solution {int ret;int target;public int findTargetSumWays(int[] nums, int _target) {target = _target;dfs(nums,0,0);return ret;}void dfs(int[] nums,int index,int sum){if(sum==target&&index == nums.length){ret++;return ;}if(index == nums.length) return ;// +dfs(nums,index+1,sum+nums[index]);// - dfs(nums,index+1,sum-nums[index]);}
}
// 只要能实现恢复现场即可,因为函数内部并没有修改sum,
// 被调函数也不会改变sum的值,因此无需恢复现场

20、组合总和★★★★根据决策树写代码

image.png
决策树一

class Solution {List<List<Integer>> ret;List<Integer> path;int sum;int aim;public List<List<Integer>> combinationSum(int[] candidates, int target) {ret = new ArrayList<>();path = new ArrayList<>();aim = target;// 先排序,否则不能利用单调性进行剪枝Arrays.sort(candidates); dfs(candidates,0);return ret;}void dfs(int[] candidates,int start){if(sum == aim){ret.add(new ArrayList<>(path));return ;}// 剪枝一:遍历的顺序从strat开始进行去重for(int i=start;i<candidates.length;i++){// 剪枝二,利用单调性可得后续的sum也大于aimif(sum+candidates[i]>aim) break; sum+=candidates[i];path.add(candidates[i]);dfs(candidates,i);// 注意此处传参为i,而不是startpath.remove(path.size()-1);sum-=candidates[i];}}
}

决策树二

枚举策略:num[i]使用多少个

class Solution {int aim;List<Integer> path;List<List<Integer>> ret;public List<List<Integer>> combinationSum(int[] nums, int target) {path = new ArrayList<>();ret = new ArrayList<>();aim = target;dfs(nums, 0, 0);return ret;}public void dfs(int[] nums, int pos, int sum) {if (sum == aim) {ret.add(new ArrayList<>(path));return;}if (sum > aim || pos == nums.length)return;// 枚举 nums[pos] 使⽤多少个for (int k = 0; k * nums[pos] + sum <= aim; k++) {if (k != 0)path.add(nums[pos]);dfs(nums, pos + 1, sum + k * nums[pos]);}// 恢复现场for (int k = 1; k * nums[pos] + sum <= aim; k++) {path.remove(path.size() - 1);}}
}

只要决策树正确,跟着决策树写代码即可,没固定模板

21、字母大小写全排列

image.png
决策树

class Solution {List<String> ret;StringBuffer path;public List<String> letterCasePermutation(String s) {ret = new ArrayList<>();path = new StringBuffer();dfs(s, 0);return ret;}void dfs(String s, int index) {if (path.length() == s.length()) {ret.add(path.toString());return;}char ch = s.charAt(index);if ('0' <= ch && ch <= '9') {path.append(ch);dfs(s,index+1);path.deleteCharAt(path.length() - 1);// 也要恢复现场}else{// ch保持不变path.append(ch);dfs(s, index + 1);path.deleteCharAt(path.length() - 1);// ch大小写转换path.append(fun(ch));dfs(s, index + 1);path.deleteCharAt(path.length() - 1);}}char fun(char c){if('a'<=c &&c<='z'){return Character.toUpperCase(c); }return Character.toLowerCase(c);}
}
// 我将所有子串分成了三类
class Solution {StringBuffer path;List<String> ret;public List<String> letterCasePermutation(String s) {path = new StringBuffer();ret = new ArrayList<>();dfs(s, 0);return ret;}public void dfs(String s, int pos) {if (pos == s.length()) {ret.add(path.toString());return;}char ch = s.charAt(pos);// 不改变path.append(ch);dfs(s, pos + 1);path.deleteCharAt(path.length() - 1); // 恢复现场// 改变if (ch < '0' || ch > '9') {char tmp = change(ch);path.append(tmp);dfs(s, pos + 1);path.deleteCharAt(path.length() - 1); // 恢复现场}}public char change(char ch) {if (ch >= 'a' && ch <= 'z')return ch -= 32;elsereturn ch += 32;}
}
// 吴老师代码是分两种情况考虑的

22、优美的排列

image.png
image.png

class Solution {int ret;boolean[] cheak;public int countArrangement(int n) {cheak = new boolean[n+1];dfs(1,n);return ret;}void dfs(int m,int n){if(m == n+1){ret++;return;}for(int i=1;i<=n;i++){if(!cheak[i]&&(i%m==0||m%i==0)){cheak[i] = true;dfs(m+1,n);cheak[i] = false;}}}
}

23、N皇后问题(以空间换时间)

image.png
决策图,只画了一半

class Solution {List<List<String>> ret;List<String> path;List<Integer> row;List<Integer> col;int n;public List<List<String>> solveNQueens(int _n) {ret = new ArrayList<>();path = new ArrayList<>();row = new ArrayList<>();col = new ArrayList<>();n = _n;dfs(0);return ret;}void dfs(int m){if(m==n){ret.add(new ArrayList<>(path));return ;}StringBuffer temp = new StringBuffer(); for(int i=0;i<n;i++) temp.append(".");for(int i=0;i<n;i++){if(m==0 || cheak(m,i)){temp.setCharAt(i,'Q');path.add(temp.toString());row.add(m);col.add(i);dfs(m+1);col.remove(col.size()-1);row.remove(row.size()-1);path.remove(path.size()-1);temp.setCharAt(i,'.');}}}boolean cheak(int x,int y){for(int i=0;i<x;i++){int a=row.get(i);int b=col.get(i);if(a-x==b-y || a-x==y-b || b == y) return false;}return true;}
}

image.png

怎么优化呢?

image.png
image.png
image.png
image.pngimage.png
总图

class Solution {List<List<String>> ret;char[][] path;// 棋盘boolean[] col;// 判断这一列是否使用过boolean[] dig1;// 判断主对角线是否使用过boolean[] dig2;// 判断副对角线是否使用过public List<List<String>> solveNQueens(int n) {ret = new ArrayList<>();path = new char[n][n];for(int i=0;i<n;i++){for(int j=0;j<n;j++){path[i][j]='.';}}col = new boolean[n];dig1 = new boolean[2*n];dig2 = new boolean[2*n];dfs(n,0);return ret;}public void dfs(int n,int y){if(y==n){List<String> list = new ArrayList<>();for(int i=0;i<n;i++){list.add(new String(path[i]));} ret.add(list);// return ;}//第index行有n个空格for(int x=0;x<n;x++){if(col[x]==false&&dig1[y-x+n]==false&&dig2[y+x]==false){path[y][x]='Q';col[x]=true;dig1[y-x+n]=true;dig2[y+x]=true;dfs(n,y+1);col[x]=false;dig1[y-x+n]=false;dig2[y+x]=false;path[y][x]='.';}}}
}

24、有效的数独 (以空间换时间)

image.png
创建布尔数组实现O(1)判断是否可以填入某数字判断是否可以填入某数字")

class Solution {boolean[][] row,col;boolean[][][] gird;public boolean isValidSudoku(char[][] board) {// 两个二维数组是为了判断某一行和某一列是否有效// 三维数组是为了判断九个九宫格是有效// 这里开辟十个空间是为了下标与数字对应row = new boolean[9][10];// 例:row[0][1]=true代表第0行存在1col = new boolean[9][10];// 例:col[0][2]=true代表第0列存在2gird = new boolean[3][3][10];// 例:gird:[0][0][4]=true代表第0个小宫存在4for(int i=0;i<9;i++){for(int j=0;j<9;j++){if(board[i][j]=='.'){continue;}int num=board[i][j]-'0';if(row[i][num]==false&&col[j][num]==false&&gird[i/3][j/3][num]==false){row[i][num]=true;col[j][num]=true;gird[i/3][j/3][num]=true;}else{return false;}}}return true;}
}

25、解数独

image.png
image.png

class Solution {char[] nums;char[][] ret = new char[9][9];public void solveSudoku(char[][] board) {nums = "123456789".toCharArray();dfs(board,0);for(int i=0;i<9;i++){for(int j=0;j<9;j++){board[i][j]=ret[i][j];}}}void dfs(char[][] board,int n){if(n==81){for(int i=0;i<9;i++){for(int j=0;j<9;j++){ret[i][j]=board[i][j];}}return;} int x = n/9;int y = n%9;if(board[x][y]!='.' ){dfs(board,n+1);}else{for(char c:nums ){if(fun(board,x,y,c)){board[x][y]=c;dfs(board,n+1);board[x][y]='.';}}}}// 判断(x,y)位置是否可以填入c,时间复杂度为O(27)boolean fun(char[][] board,int x,int y,char c){int m = x/3;int n = y/3;for(int i=3*m;i<3*m+3;i++){for(int j=3*n;j<3*n+3;j++){if(board[i][j]==c) return false;}}for(int i=0;i<9;i++){if(board[x][i]==c) return false;if(board[i][y]==c) return false;}return true;}
}
class Solution {//直接修改二维数组boolean[][] col,row;boolean[][][] grid;char[][] temp;boolean flag=false;public void solveSudoku(char[][] board) {// 三兄弟在填写的时候用于判断某个数字是否合法,添加数字后需要更三兄弟,还要牵扯到回溯col = new boolean[9][10];row = new boolean[9][10];grid = new boolean[3][3][10];temp = new char[9][9];//初始化一下三兄弟for(int i=0;i<9;i++){for(int j=0;j<9;j++){if(board[i][j]!='.'){int num = board[i][j]-'0';row[i][num]=true;col[j][num]=true;grid[i/3][j/3][num]=true;}}}// (0,0)下标开始。这参数设置的不如我的n,当n==81时,说明找到了答案dfs(board,0,0);// for(int s=0;s<9;s++){//     for(int k=0;k<9;k++){//         board[s][k]=temp[s][k];//     }// }}public void dfs(char[][] board,int i,int j){if(i==9){// 如何保存返回值,如果不进行回溯的话,board数组会在回溯的变成初始的样子// 保存返回值的第一种方法就是对结果进行复制 // for(int s=0;s<9;s++){//     for(int k=0;k<9;k++){//         temp[s][k]=board[s][k];//     }// }flag = true;}// 题目保证有且仅有一个解,所以当找到答案时可以直接一路返回无需回溯if(flag) return;// 已找到答案,所以returnif(board[i][j]=='.'){for(int k=1;k<=9;k++){if(row[i][k]==false&&col[j][k]==false&&grid[i/3][j/3][k]==false){board[i][j]=(char)(48+k);row[i][k]=true;col[j][k]=true;grid[i/3][j/3][k]=true;if(j==8){dfs(board,i+1,0);}else{dfs(board,i,j+1);}// 题目保证有且仅有一个解,所以当找到答案时可以直接一路返回无需回溯if(flag) return;// 防止回溯,所以return// 回溯操作board[i][j]='.' ;row[i][k]=false;col[j][k]=false;grid[i/3][j/3][k]=false;}}    }else{if(j==8){dfs(board,i+1,0);}else{dfs(board,i,j+1);}}}
}

26、单词搜索 (dfs只能解决以某个起点为开始的问题,所以循环调用dfs)

image.png
image.png

向量数组

class Solution {boolean ret;int[] dx = {0,0,-1,1};int[] dy = {-1,1,0,0};int m;int n;public boolean exist(char[][] board, String word) {m = board.length;n = board[0].length;boolean[][] cheak = new boolean[m][n]; for(int i=0;i<m;i++){for(int j=0;j<n;j++){// 每次dfs都会将cheak回溯到初始状态,不影响下一次dfsdfs(board,i,j,word,0,cheak);if(ret) break;}}return ret;}void  dfs(char[][] board,int x,int y,String word,int index,boolean[][] cheak){if(ret) return ;if(cheak[x][y]==false && board[x][y]==word.charAt(index)){cheak[x][y] = true;index++;if(index==word.length()){ret = true;return ;}for(int i=0;i<4;i++){int a = x+dx[i];int b = y+dy[i];if(a>=0&&a<m&&b>=0&&b<n&&cheak[a][b]==false)dfs(board,a,b,word,index,cheak);}// 没有在回溯之前返回,所以会回溯if(ret) return;cheak[x][y] = false;index--;}}
}
class Solution {boolean flag=false;public boolean exist(char[][] board, String word) {int m=board.length;int n=board[0].length;char[][] temp = new char[m+2][n+2];boolean[][] cheak = new boolean[m+2][n+2];for(int i=0;i<m+2;i++){for(int j=0;j<n+2;j++){if(i==0||j==0||i==m+1||j==n+1){temp[i][j]='0';cheak[i][j]=true;}else{temp[i][j]=board[i-1][j-1];cheak[i][j]=false;}}}char[] arr = word.toCharArray();for(int i=1;i<=m+1;i++){for(int j=1;j<=n+1;j++){dfs(temp,0,arr,i,j,cheak);if(flag) return true;}}return flag;}public void dfs(char[][] temp,int index,char[] arr,int row,int col,boolean[][] cheak){int m=temp.length;int n=temp[0].length;if(index==arr.length){flag=true;return ;}if(flag) return;if(cheak[row][col]==true){return ;}if(temp[row][col]==arr[index]){cheak[row][col]=true;dfs(temp,index+1,arr,row-1,col,cheak);dfs(temp,index+1,arr,row,col+1,cheak);dfs(temp,index+1,arr,row+1,col,cheak);dfs(temp,index+1,arr,row,col-1,cheak);cheak[row][col]=false;return ;}}
}  
class Solution {boolean[][] vis;int m, n;char[] word;public boolean exist(char[][] board, String _word) {m = board.length;n = board[0].length;word = _word.toCharArray();vis = new boolean[m][n];for (int i = 0; i < m; i++)for (int j = 0; j < n; j++) {if (board[i][j] == word[0]) {vis[i][j] = true;if (dfs(board, i, j, 1))return true;vis[i][j] = false;}}return false;}int[] dx = { 0, 0, 1, -1 };int[] dy = { 1, -1, 0, 0 };public boolean dfs(char[][] board, int i, int j, int pos) {if (pos == word.length) {return true;}// 上下左右去匹配 word[pos]// 利⽤向量数组,⼀个 for 搞定上下左右四个⽅向for (int k = 0; k < 4; k++) {int x = i + dx[k], y = j + dy[k];if (x >= 0 && x < m && y >= 0 && y < n && !vis[x][y] && board[x][y] == word[pos]) {vis[x][y] = true;if (dfs(board, x, y, pos + 1))return true;vis[x][y] = false;}}return false;}
}

27、黄金矿工★★★★

image.png
从6位置开始挖矿的深度优先搜索图

class Solution {int m;int n;public int getMaximumGold(int[][] grid) {m = grid.length;n = grid[0].length;int max = 0;// dfs只能实现从一个坐标开始深度搜索找到从这个起点开始的最大值// 因此要二重循环调用dfs取最大值for(int i=0;i<m;i++){for(int j=0;j<n;j++){if(grid[i][j]!=0)max = Math.max(max,dfs(grid,i,j));}}return max;}// 向量数组int[] dx = { 0, 0, 1, -1 };int[] dy = { 1, -1, 0, 0 };// 从某个起点开始找最大值int dfs(int[][] grid, int x, int y) {int max = 0;for (int i = 0; i < 4; i++) {int a = x + dx[i];int b = y + dy[i];if (a >= 0 && a < m && 0 <= b && b < n && grid[a][b]!=0){int temp = grid[x][y];grid[x][y]=0;// 不要错写成grid[a][b];max = Math.max(max,dfs(grid, a, b));grid[x][y]=temp;}}return grid[x][y] + max;}
}

28、不同路径III

image.png
该矩阵中有两条路径可以从起点走到终点并遍布所有的0位置

class Solution {int ret;int m;int n;int[] start;// 记录起点int[] end;// 记录终点int path=1;// 保证走遍所有的0public int uniquePathsIII(int[][] grid) {m =  grid.length;n =  grid[0].length;for(int i=0;i<m;i++){for(int j=0;j<n;j++){if(grid[i][j]==1) start = new int[]{i,j};if(grid[i][j]==2) end = new int[]{i,j};if(grid[i][j]==0) path++;}}dfs(grid,start[0],start[1]);return ret;}int[] dx = {0,0,1,-1};int[] dy = {1,-1,0,0};void dfs(int[][] grid,int x,int y){if(end[0]==x&&end[1]==y&&path==0){ret++;return ;} for(int i=0;i<4;i++){int a = x+dx[i];int b = y+dy[i];if(0<=a&&a<m&&0<=b&&b<n&&grid[a][b]!=-1){int temp = grid[x][y];grid[x][y] = -1;path--;dfs(grid,a,b);path++;grid[x][y] = temp;}}}}

五、FloodFill 算法

前四章总结
第五章介绍

29、图像渲染

image.png

class Solution {public int[][] floodFill(int[][] image, int sr, int sc, int color) {// 宽度优先遍历int prev = image[sr][sc];if (prev == color)return image;// 方向int[] di = new int[] { 0, 0, 1, -1 };int[] dj = new int[] { 1, -1, 0, 0 };// 队列Queue<int[]> queue = new LinkedList<>();queue.add(new int[] { sr, sc });int m = image.length, n = image[0].length;while (!queue.isEmpty()) {int[] arr = queue.poll();int x = arr[0], y = arr[1];image[x][y] = color;for (int i = 0; i < 4; i++) {int a = x + di[i];int b = y + dj[i];if (0 <= a && a < m && 0 <= b && b < n && image[a][b] == prev) {queue.add(new int[] { a, b });}}}return image;}
}
class Solution {int[] dx = {0,0,1,-1};int[] dy = {1,-1,0,0};public int[][] floodFill(int[][] image, int sr, int sc, int color) {int oldColor = image[sr][sc];if(oldColor == color) return image; // 不能没有int m = image.length;int n = image[0].length;image[sr][sc] = color;for(int i=0;i<4;i++){int x = sr+dx[i];int y = sc+dy[i];if(0<=x&&x<m&&0<=y&&y<n&&image[x][y]==oldColor){floodFill(image,x,y,color);}}return image;}
}

DFS vs BFS:

bfs:四个方向同时一层一层的往外走dfs:四个方向一条路走到黑

30、岛屿数量

image.png

class Solution {public int numIslands(char[][] grid) {// 加强版图像渲染// 引入一个二维数组用于标记是否已经探索过,可以直接修改元素组,一般都还是建立一个标记数组int m = grid.length;int n = grid[0].length;int[] dx = new int[] { 0, 0, 1, -1 };int[] dy = new int[] { 1, -1, 0, 0 };boolean[][] flag = new boolean[m][n];for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {flag[i][j] = false;}}// 创建队列;Queue<int[]> queue = new LinkedList<>();int ret = 0;for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {if (grid[i][j] == '1' && flag[i][j] == false) {queue.add(new int[] { i, j });flag[i][j] = true;while (!queue.isEmpty()) {int[] temp = queue.poll();int x = temp[0], y = temp[1];for (int k = 0; k < 4; k++) {int a = x + dx[k], b = y + dy[k];if (0 <= a && a < m && 0 <= b && b < n && grid[a][b] == '1' && flag[a][b] == false) {queue.add(new int[] { a, b });flag[a][b] = true;}}}ret++;}}}return ret;}
}
class Solution {int m;int n;public int numIslands(char[][] grid) {m = grid.length;n = grid[0].length;int ret=0;for(int i=0;i<m;i++){for(int j=0;j<n;j++){if(grid[i][j]=='1'){dfs(grid,i,j);ret++;} }}return ret;}int[] dx = { 0, 0, 1, -1 };int[] dy = { 1, -1, 0, 0 };void dfs(char[][] grid, int x, int y) {// 标记这个位置已经蔓延过来,防止折返grid[x][y]='0';for (int i = 0; i < 4; i++) {int a = x + dx[i];int b = y + dy[i];// 蔓延条件:坐标合法,性质相同if (0 <= a && a < m && 0 <= b && b < n && grid[a][b]=='1') {dfs(grid,a,b);}}}
}

31、最大岛屿面积

image.png

class Solution {public int maxAreaOfIsland(int[][] grid) {// 加强版图像渲染// 引入一个二维数组用于标记是否已经探索过,可以直接修改元素组,一般都还是建立一个标记数组int m = grid.length;int n = grid[0].length;int[] dx = new int[] { 0, 0, 1, -1 };int[] dy = new int[] { 1, -1, 0, 0 };boolean[][] flag = new boolean[m][n];for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {flag[i][j] = false;}}// 创建队列;Queue<int[]> queue = new LinkedList<>();int max = 0;for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {if (grid[i][j] == 1 && flag[i][j] == false) {queue.add(new int[] { i, j });flag[i][j] = true;int size=0;while (!queue.isEmpty()) {int[] temp = queue.poll();size++;int x = temp[0], y = temp[1];for (int k = 0; k < 4; k++) {int a = x + dx[k], b = y + dy[k];if (0 <= a && a < m && 0 <= b && b < n && grid[a][b] == 1 && flag[a][b] == false) {queue.add(new int[] { a, b });flag[a][b] = true;}}}max = Math.max(max,size);}}}return max;}
}
class Solution {int m;int n;int[] dx = { 0, 0, 1, -1 };int[] dy = { 1, -1, 0, 0 };public int maxAreaOfIsland(int[][] grid) {m = grid.length;n = grid[0].length;int max = 0;for(int i=0;i<m;i++)for(int j=0;j<n;j++)if(grid[i][j]==1)max = Math.max(max,dfs(grid,i,j));return max;}int dfs(int[][] grid, int x, int y) {// 记录岛屿大小int ret = 1; // 标记这个位置已经蔓延过了,防止折返grid[x][y] = 0;for (int i = 0; i < 4; i++) {int a = x + dx[i];int b = y + dy[i];// 向四周继续蔓延的条件:坐标合法,性质相同if (0 <= a && a < m && 0 <= b && b < n && grid[a][b] == 1) {ret += dfs(grid, a, b);}}return ret;}
}

代码对DFS进行了修改,如果不想修改DFS可以添加一个布尔数组,比较遍历过的地方,防止重复遍历

32、被围绕的区域

image.png

class Solution {// 定义在里面的需要手动初始化int[] dx = new int[] { 0, 0, 1, -1 };int[] dy = new int[] { 1, -1, 0, 0 };int m, n;Queue<int[]> q = new LinkedList<>();public void solve(char[][] board) {m = board.length;n = board[0].length;// 1. 先处理边界的 'O' 联通块,全部修改成 '.'for (int j = 0; j < n; j++) {if (board[0][j] == 'O') {bfs2(board, 0, j);}if (board[m - 1][j] == 'O') {bfs2(board, m - 1, j);}}// 1. 也是先处理边界的 'O' 联通块,全部修改成 '.'for (int i = 0; i < m; i++) {if (board[i][0] == 'O') {bfs2(board, i, 0);}if (board[i][n - 1] == 'O') {bfs2(board, i, n - 1);}}// 构造返回值for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {if (board[i][j] == 'O') {board[i][j] = 'X';}if (board[i][j] == '.') {board[i][j] = 'O';}}}}public void bfs(char[][] board, int i, int j) {q.add(new int[] { i, j });while (!q.isEmpty()) {int[] t = q.poll();int a = t[0], b = t[1];board[a][b] = '.';for (int k = 0; k < 4; k++) {int x = a + dx[k], y = b + dy[k];if (x >= 0 && x < m && y >= 0 && y < n && board[x][y] == 'O') {q.add(new int[] { x, y });}}}}public void bfs2(char[][] board, int i, int j) {board[i][j] = '.';q.add(new int[] { i, j });while (!q.isEmpty()) {int[] temp = q.poll();int x = temp[0], y = temp[1];for (int k = 0; k < 4; k++) {int a = x + dx[k], b = y + dy[k];if (0 <= a && a < m && 0 <= b && b < n && board[a][b] == 'O') {board[a][b] = '.';q.add(new int[] { a, b });}}}}}
class Solution {// 定义在里面的需要手动初始化int[] dx = new int[] { 0, 0, 1, -1 };int[] dy = new int[] { 1, -1, 0, 0 };int m, n;Queue<int[]> q = new LinkedList<>();public void solve(char[][] board) {m = board.length;n = board[0].length;// 1. 先处理边界的 'O' 联通块,全部修改成 '.'for (int j = 0; j < n; j++) {if (board[0][j] == 'O') {dfs(board, 0, j);}if (board[m - 1][j] == 'O') {dfs(board, m - 1, j);}}// 1. 也是先处理边界的 'O' 联通块,全部修改成 '.'for (int i = 0; i < m; i++) {if (board[i][0] == 'O') {dfs(board, i, 0);}if (board[i][n - 1] == 'O') {dfs(board, i, n - 1);}}// 构造返回值for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {if (board[i][j] == 'O') {board[i][j] = 'X';}if (board[i][j] == '.') {board[i][j] = 'O';}}}}void dfs(char[][] board ,int x,int y){board[x][y] = '.';for (int i = 0; i < 4; i++) {int a = x + dx[i];int b = y + dy[i];if (0 <= a && a < m && 0 <= b && b < n && board[a][b] == 'O') {dfs(board, a, b);}}}
}

只是讲BFS函数改为了DFS,solve函数的主体不变

33、太平洋大西洋水流问题 (正南则反的典型案例)

image.png

class Solution {int[] dx = new int[] { 0, 0, 1, -1 };int[] dy = new int[] { 1, -1, 0, 0 };int m, n;boolean pacific;boolean atlantic;List<List<Integer>> ret;boolean[][] cheak;public List<List<Integer>> pacificAtlantic(int[][] heights) {ret = new ArrayList<>();m = heights.length;n = heights[0].length;cheak = new boolean[m][n];for(int i=0;i<m;i++){for(int j=0;j<n;j++){pacific = false;atlantic = false;dfs(heights,i,j);if(pacific && atlantic){List<Integer> arr = new ArrayList<>();arr.add(i);arr.add(j);ret.add(arr);}}}return ret;}// 判断从某个位置开始是否可以流入大西洋和太平洋void dfs(int[][] heights,int x,int y){if(x==0||y==0) pacific = true;if(x==m-1||y==n-1) atlantic = true;if(pacific && atlantic) return;for (int i = 0; i < 4; i++) {int a = x + dx[i];int b = y + dy[i];if (0 <= a && a < m && 0 <= b && b < n && heights[x][y]>=heights[a][b] && !cheak[x][y]) {cheak[x][y] = true;dfs(heights, a, b);cheak[x][y] = false;}}}}

思路一:设计dfs函数用于判断某一个点是否可以流入两个大洋,循环遍历每一个位置

class Solution {int m, n;int[] dx = { 0, 0, 1, -1 };int[] dy = { 1, -1, 0, 0 };public List<List<Integer>> pacificAtlantic(int[][] h) {m = h.length;n = h[0].length;boolean[][] pac = new boolean[m][n];boolean[][] atl = new boolean[m][n];// 1. 先处理 pac 洋for (int j = 0; j < n; j++)dfs(h, 0, j, pac);for (int i = 0; i < m; i++)dfs(h, i, 0, pac);// 2. 再处理 atl 洋for (int i = 0; i < m; i++)dfs(h, i, n - 1, atl);for (int j = 0; j < n; j++)dfs(h, m - 1, j, atl);// 3. 提取结果List<List<Integer>> ret = new ArrayList<>();for (int i = 0; i < m; i++)for (int j = 0; j < n; j++)if (pac[i][j] && atl[i][j]) {List<Integer> tmp = new ArrayList<>();tmp.add(i);tmp.add(j);ret.add(tmp);}return ret;}public void dfs(int[][] h, int i, int j, boolean[][] vis) {vis[i][j] = true;for (int k = 0; k < 4; k++) {int x = i + dx[k], y = j + dy[k];if (x >= 0 && x < m && y >= 0 && y < n && !vis[x][y] && h[x][y] >= h[i][j]) {dfs(h, x, y, vis);}}}
}

思路二:正难则反,从沿岸开始往上爬。分别从两个海岸往上爬,从两边都能爬到的地方就是答案。

34、扫雷问题

image.png

class Solution {int[] dx = {0,0,1,-1,1,-1,1,-1};int[] dy = {1,-1,0,0,1,-1,-1,1};int m,n;public char[][] updateBoard(char[][] board, int[] click) {m = board.length;n = board[0].length;int x = click[0];int y = click[1];if(board[x][y]=='M'){board[x][y]='X';return board;}else if(board[x][y]=='E'){dfs(board,x,y);return board;}else{return board;}    }void dfs(char[][] board,int x,int y){int k = 0;// 更具题目要求做事情:for(int i=0;i<8;i++){int a = x+dx[i];int b = y+dy[i];if(0<=a&&a<m&&0<=b&&b<n&&board[a][b]=='M') k++;}if(k>0){board[x][y] = (char)('0'+k);return ;}board[x][y] = 'B';for(int i=0;i<8;i++){int a = dx[i]+x;int b = dy[i]+y;if(0<=a&&a<m&&0<=b&&b<n&&board[a][b]=='E'){dfs(board,a,b);}}    }}

点击到炸弹就在updateBoard中解决,点击到空格在dfs中解决

35、衣柜整理

image.png

class Solution {int m, n, cnt;int ret;boolean[][] cheak;public int wardrobeFinishing(int _m, int _n, int _cnt) {m = _m;n = _n;cheak = new boolean[m][n];cnt = _cnt;dfs(0, 0);return ret;}int[] dx = { 0, 1 };int[] dy = { 1, 0 };void dfs(int x, int y) {// if (fun(x) + fun(y) <= cnt) {ret++;// 标记已经遍历的地方cheak[x][y] = true;for (int i = 0; i < 2; i++) {int a = dx[i] + x;int b = dy[i] + y;// 继续搜索的条件:坐标合法,未被搜索过if (0 <= a && a < m && 0 <= b && b < n && !cheak[a][b])dfs(a, b);}}}int fun(int x) {int count = 0;while (x > 0) {count += x % 10;x /= 10;}return count;}
}
/** 0123* 1234* 2345* 345* 45* 5* 6* 7*/

六、 记忆化搜索

36、斐波那契数

image.png递归解法,存在重复子问题
image.png
动态规划与记忆化搜索的关系,本质是一样的
image.pngimage.png

记忆化搜索与动态规划本质都是对暴搜的优化,只不过一个是递归形式,一个是循环形式

class Solution {public int fib(int n) {return dfs(n);}public int dfs(int n){if(n==0) return 0;if(n==1) return 1;return dfs(n-1)+dfs(n-2);}
}
class Solution {int[] memo = new int[31];public int fib(int n) {// 初始化备忘录for(int i=0;i<31;i++) memo[i]=-1;return dfs(n);}int dfs(int n) {if (memo[n] != -1){return memo[n]; // 直接去备忘录⾥⾯拿值}if (n == 0 || n == 1) {memo[n] = n; // 记录到备忘录⾥⾯return n;}memo[n] = dfs(n - 1) + dfs(n - 2); // 记录到备忘录⾥⾯return memo[n];}
}
class Solution {public int fib(int n) {if(n<=1) return n;int[] dp = new int[n+1];dp[0]=0;dp[1]=1;for(int i=2;i<=n;i++){dp[i]=dp[i-1]+dp[i-2];}return dp[n];}
}

记忆化搜索与动态规划本质都是对暴搜的优化,只不过一个是递归形式,一个是循环形式

37、不同路径

image.png

// 我的代码,自底向上
class Solution {int[][] memo ;int m,n;public int uniquePaths(int _m, int _n) {m = _m;n = _n;memo = new int[m+1][n+1];return dfs(1,1);}int dfs(int x,int y){if(x>m||y>n) return 0;if(memo[x][y]!=0) return memo[x][y];// if((x==m && y==n)||(x==m-1 && y==n)||(x==m && y==n-1)){memo[x][y]=1;return 1;} memo[x][y] = dfs(x+1,y)+dfs(x,y+1);// 这里忘记写了return memo[x][y];}
}// 吴老师的代码,自顶向下 
class Solution {public int uniquePaths(int m, int n) {// 记忆化搜索int[][] memo = new int[m + 1][n + 1];return dfs(m, n, memo);}public int dfs(int i, int j, int[][] memo) {if (memo[i][j] != 0) {return memo[i][j];}if (i == 0 || j == 0)return 0;if (i == 1 && j == 1) {memo[i][j] = 1;return 1;}memo[i][j] = dfs(i - 1, j, memo) + dfs(i, j - 1, memo);return memo[i][j];}
}

38、最长递增子序列

image.png

画决策树有助于写暴搜

以某个位置为起点的递增子序列暴搜索示意图

class Solution {public int lengthOfLIS(int[] nums) {int max = 0;// 循环求出以每一个位置为起点的最长递增子序列,然后取最大值for(int i=0;i<nums.length;i++){max = Math.max(max,dfs(nums,i));}return max;}// dfs:以index为起点的最长递增子序列int dfs(int[] nums,int index){int max = 0;for(int i=index+1;i<nums.length;i++){if(nums[i]>nums[index])max = Math.max(max,dfs(nums,i));}return 1+max;}
}

如何改成记忆化搜:

  1. 首先要确定重复子问题是什么
  2. 备忘录的初始化,如果备忘录中有结果直接从备忘录中返回即可
  3. 填写备忘录,将所有dfs的返回值都更新到备忘录中
    1. 有些是可以直接填写相当于递归出口或者是动态规划中dp表的初始化
    2. 有些是递归调用得出来的及结果

对于 nums={1,2,3,4,5},dfs(nums,1)和dfs(nums,2)都会调用dfs(nums,3),因此在计算dfs(nums,1)的时候会把dfs(nums,3)计算出来,将dfs(nums,3)的结果写入备忘录,当dfs(nums,2)掉用dfs(nums,3)时直接查找备忘录即可。

class Solution {int[] memo;public int lengthOfLIS(int[] nums) {memo = new int[nums.length];int max = 0;for(int i=0;i<nums.length;i++){max = Math.max(max,dfs(nums,i));}return max;}// dfs:以index为起点的最长递增子序列int dfs(int[] nums,int index){if(memo[index]!=0) return memo[index];// 此处相当于递归出口,或者是动态规划中dp表的初始化if(index==nums.length-1){memo[index]=1;return memo[index];}int max = 0;for(int i=index+1;i<nums.length;i++){if(nums[i]>nums[index]){memo[i] = dfs(nums,i);max = Math.max(max,memo[i]);}}memo[index]=1+max;return memo[index];}
}
class Solution {public int lengthOfLIS(int[] nums) {int n = nums.length;// dp[i]:以i位置为结尾的最长递增子序列的长度int[] dp = new int[n];dp[0] = 1;int max =dp[0];for(int i=1;i<n;i++){dp[i] = 1;for(int j=i-1;j>=0;j--){if(nums[i]>nums[j]){dp[i] = Math.max(dp[j]+1,dp[i]);}}max = Math.max(max,dp[i]);}return max;}
}
// 这一题的最优解是贪心策略

39、猜数字大小II ★★★★★(有趣,难)

image.png
image.png

上图第二课决策树能保证获胜的钱数为5+7+8+9=29;

class Solution {public int getMoneyAmount(int n) {// 根据 dfs 的含义可知,dfs(1,n)就是题目的答案return dfs(1,n);}// 保证游戏获胜的最小钱数游戏范围是[left,right];int dfs(int left,int right){// 两个特殊情况if(left == right) return 0;if(left+1==right) return left;int ret = Integer.MAX_VALUE;// 循环讨论,列举所有决策树// 左子树:[left,i-1] // 根节点: i// 右子树:[i+1,right]// 根据这个棵树保证能获胜的钱数为:根节点+左右子树的大者for(int i = left+1;i<right;i++){int temp = i + Math.max(dfs(left,i-1),dfs(i+1,right));ret = Math.min(ret,temp);}return ret;}
}

该问题存在重复子问题:dfs(a+b)和dfs(a,b-1)都会去调用dfs(a,i-1);
使用二维数组充当备忘录,memo[left,right]记录当猜数范围是[left,right]时的保证能获胜的最小金额

class Solution {int[][] memo;public int getMoneyAmount(int n) {memo = new int[n+1][n+1];return dfs(1,n);}int dfs(int left,int right){if(memo[left][right]!=0) return memo[left][right];if(left == right){memo[left][right]=0;return 0;} if(left+1==right) {memo[left+1][right] = 0;return left;}int ret = Integer.MAX_VALUE;for(int i = left+1;i<right;i++){// 注意这里要用max,(dfs(left,i-1),dfs(i+1,right))是左右子树所需的最小值,// 根节点所需的最小值为i+Math.max(dfs(left,i-1),dfs(i+1,right))memo[left][i-1] = dfs(left,i-1);memo[i+1][right] = dfs(i+1,right);int temp = i+Math.max(memo[left][i-1],memo[i+1][right]);ret = Math.min(ret,temp);}memo[left][right]=ret;return ret;}
}

40、矩阵的最长递增路径

image.png

class Solution {int m,n;public int longestIncreasingPath(int[][] matrix) {m = matrix.length;n = matrix[0].length;int max = 0;for(int i=0;i<m;i++){for(int j=0;j<n;j++){// 循环计算以每一个位置为起点的最长递增路径,然后取最大值max = Math.max(max,dfs(matrix,i,j));}}return max;}int[] dx = {0,0,1,-1};int[] dy = {1,-1,0,0};// 求解的是以某一个位置为起点的最长递增路径int dfs(int[][] matrix,int x,int y){int max=0;for(int i=0;i<4;i++){int a = dx[i]+x;int b = dy[i]+y;if(0<=a&&a<m&&0<=b&&b<n&&matrix[a][b]>matrix[x][y]){max = Math.max(max,dfs(matrix,a,b));}}return 1+max;}}

从任意位置(x,y)开始沿着递增的方向深度优先搜索,记录路径的长度

class Solution {int m,n;int[][] memo;public int longestIncreasingPath(int[][] matrix) {m = matrix.length;n = matrix[0].length;memo = new int[m][n];int max = 0;for(int i=0;i<m;i++){for(int j=0;j<n;j++){max = Math.max(max,dfs(matrix,i,j));}}return max;}int[] dx = {0,0,1,-1};int[] dy = {1,-1,0,0};// 题目存在重复子问题——>改成记忆化搜索的形式// 使用二维数组充当备忘录,将dfs(a,b)存储在memo[a][b]中,// 如果dfs[x][y]或者dfs[i][j]需要调用dfs(a,b)的时候直接查找memo即可int dfs(int[][] matrix,int x,int y){if(memo[x][y]!=0) return memo[x][y];int max=0;// 没有出口for(int i=0;i<4;i++){int a = dx[i]+x;int b = dy[i]+y;if(0<=a&&a<m&&0<=b&&b<n&&matrix[a][b]>matrix[x][y]){memo[a][b] = dfs(matrix,a,b); // 将所有dfs()的结果补充到备忘录中max = Math.max(max,memo[a][b]);}}memo[x][y] = 1+max;// 将所有dfs()的结果补充到备忘录中return 1+max;}}

dfs函数示意图
dfs(matrix,x,y)从(x,y)位置开始像四个方向蔓延,然后取这四个方向递增路径的最大值,因为只是算出了以某位置为起点的最长递增路径长度,所以主函数要循环调用dfs来及计算以每一个位置为起点的最长递增路径,然后取最大值。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/43971.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Python | Leetcode Python题解之第226题翻转二叉树

题目&#xff1a; 题解&#xff1a; class Solution:def invertTree(self, root: TreeNode) -> TreeNode:if not root:return rootleft self.invertTree(root.left)right self.invertTree(root.right)root.left, root.right right, leftreturn root

01_空中机器人

空中机器人&#xff08;Aerial Robotics&#xff09;最早由美国乔治亚理工大学的Robert Michelson提出&#xff0c;是指各种搭载了GPS、机载导航设备、视觉识别设备以及无线通信设备等&#xff0c;能够在一定的范围内实现无人飞行的旋翼无人飞行器、无人飞艇等。 空中机器人拓…

Zynq系列FPGA实现SDI视频编解码+图像缩放+多路视频拼接,基于GTX高速接口,提供8套工程源码和技术支持

目录 1、前言工程概述免责声明 2、相关方案推荐本博已有的 SDI 编解码方案本博已有的FPGA图像缩放方案本方案的无缩放应用本方案在Xilinx--Kintex系列FPGA上的应用 3、详细设计方案设计原理框图SDI 输入设备Gv8601a 均衡器GTX 解串与串化SMPTE SD/HD/3G SDI IP核BT1120转RGB自研…

14-58 剑和诗人32 - 使用矢量数据库增强 LLM 应用程序

GPT-4、Bloom、LaMDA 等大型语言模型 (LLM) 在生成类似人类的文本方面表现出了令人印象深刻的能力。然而,它们在事实准确性和推理能力等方面仍然面临限制。这是因为,虽然它们的基础是从大量文本数据中提取统计模式,但它们缺乏结构化的知识源来为其输出提供依据。 最近,我们…

基于信号量的生产者消费者模型

文章目录 信号量认识概念基于线程分析信号量信号量操作 循环队列下的生产者消费者模型理论认识代码部分 信号量 认识概念 信号量本质: 计数器 它也叫做公共资源 为了线程之间,进程间通信------>多个执行流看到的同一份资源---->多个资源都会并发访问这个资源(此时易出现…

【Linux】进程(9):进程控制2(进程等待)

大家好&#xff0c;我是苏貝&#xff0c;本篇博客带大家了解Linux进程&#xff08;9&#xff09;进程控制2&#xff0c;如果你觉得我写的还不错的话&#xff0c;可以给我一个赞&#x1f44d;吗&#xff0c;感谢❤️ 目录 一. 为什么要进程等待二. 如何进行进程等待1.wait函数—…

使用linux的mail命令发送html格式的邮件

1、关闭本机的sendmail服务或者postfix服务 #执行下面的命令&#xff0c;各位大侠都对号入座吧 #sendmial service sendmail stop chkconfig sendmail off #postfix service postfix stop chkconfig postfix off#再狠一点就直接卸载吧.. yum remove sendmail yum remove postf…

欧拉部署nginx

1.下载nginx 下载地址&#xff1a;https://nginx.org/en/download.html 选择稳定版本 下的镜像文件进行下载 2.解压Nginx包 cd /root/nginx tar -zxvf nginx-1.26.0.tar.gz cd nginx-1.26.03.安装nginx相关依赖 yum -y install gcc zlib zlib-devel pcre-devel openssl o…

如何在 CentOS 中配置 Linux 命名空间(ip netns)

引言 Linux 命名空间是一项强大的技术&#xff0c;允许在同一系统上创建多个独立的虚拟化实例&#xff0c;每个实例可以拥有自己的网络栈、路由表、IP 地址等网络资源&#xff0c;实现资源的隔离和管理。本文将深入探讨如何在 CentOS 中配置和使用 ip netns 命名空间&#xff0…

【面试题】正向代理和反向代理的区别?

正向代理&#xff08;Forward Proxy&#xff09;和反向代理&#xff08;Reverse Proxy&#xff09;是两种常见的代理服务器类型&#xff0c;它们在网络通信中扮演着不同的角色&#xff0c;具有不同的功能和应用场景。 一、正向代理 1. 定义与位置 正向代理是位于客户端和目标…

TextView 实现最后一行缩进指定距离

实现图上类似的效果。 指定最大行数为三行&#xff0c;最后一行缩进指定的距离。 如果行数小于三行&#xff0c;则不缩进。 同时文字两端对齐 代码里的 JustifyTextView &#xff08;两端对齐的 Textview &#xff09;详见 Android Textview 多行文本两端对齐_android tex…

Go语言入门之基础语法

Go语言入门之基础语法 1.简单语法概述 行分隔符&#xff1a; 一行代表一个语句结束&#xff0c;无需写分号。将多个语句写在一行可以用分号分隔&#xff0c;但是不推荐 注释&#xff1a; // 或者/* */ 标识符&#xff1a; 用来命名变量、类型等程序实体。 支持大小写字母、数字…

k8s核心操作_Deployment的扩缩容能力_Deployment自愈和故障转移能力---分布式云原生部署架构搭建022

然后我们上面说了k8s中的deployment的多副本能力 然后,我们再来看 k8s中的deployment的扩缩容能力 可以看到,对于扩容,要使用 kubectl scale 命令 对于缩容 要使用kubectl scale 命令都是使用这个命令对吧 来试试,可以看到上面命令 首先看看 kubectl get pod 可以看到有…

第58期 | GPTSecurity周报

GPTSecurity是一个涵盖了前沿学术研究和实践经验分享的社区&#xff0c;集成了生成预训练Transformer&#xff08;GPT&#xff09;、人工智能生成内容&#xff08;AIGC&#xff09;以及大语言模型&#xff08;LLM&#xff09;等安全领域应用的知识。在这里&#xff0c;您可以找…

网络编程:TCP

一、tcp编程 注意 1.数据本身有顺序 2.发送和接收次数不需要对应 3. 1. C/S 模式 》服务器/客户端模型 server:socket()-->bind()--->listen()-->accept()-->recv()-->close() client:socket()-->connect()-->send()-->close(); int on 1; setso…

常用的设计模式和使用案例汇总

常用的设计模式和使用案例汇总 【一】常用的设计模式介绍【1】设计模式分类【2】软件设计七大原则(OOP原则) 【二】单例模式【1】介绍【2】饿汉式单例【3】懒汉式单例【4】静态内部类单例【5】枚举&#xff08;懒汉式&#xff09; 【三】工厂方法模式【1】简单工厂模式&#xf…

GuLi商城-商品服务-API-品牌管理-OSS获取服务端签名

新建第三方服务: 引入common 把common中oss的依赖都拿到第三方服务中来 配置文件: 加上nacos注解:<

HTML 标签简写和全称及其对应的中文说明和实例

<!DOCTYPE html> <html lang"zh-CN"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>HTML 标签简写及全称</title><style>…

Android 通知访问权限

问题背景 客户反馈手机扫描三方运动手表&#xff0c;下载app安装后&#xff0c;通知访问权限打不开。 点击提示“受限设置” “出于安全考虑&#xff0c;此设置目前不可用”。 问题分析 1、setting界面搜“授予通知访问权限”&#xff0c;此按钮灰色不可点击&#xff0c;点…

大小端详解

引例 我们知道整形(int)是4个字节&#xff0c;例如随便举个例子&#xff1a;0x01020304&#xff0c;它一共占了四个地址位&#xff0c;01,02,03,04分别占了一个字节&#xff08;一个字节就对应了一个地址&#xff09;。 那么就会有个问题&#xff1a;我们的01到底是存储在高地…