你要开发一座金矿,地质勘测学家已经探明了这座金矿中的资源分布,并用大小为 m * n
的网格grid
进行了标注。每个单元格中的整数就表示这一单元格中的黄金数量;如果该单元格是空的,那么就是0
。
为了使收益最大化,矿工需要按以下规则来开采黄金:
0
的单元格。遍历每一个格子,如果当前正在遍历的格子的值不是0
,使用DFS
搜索一遍答案,当到底时,更新答案。回溯时需要还原现场,因此需要将之前置为true
的格子重新置为false
,这样才能保证所有的路径都能够被遍历到。(区别于求岛屿最大面积)。
class Solution {
public static int ans;
public static boolean[][] isVisited;
public static int[] dX = {0, 0, 1, -1};
public static int[] dY = {1, -1, 0, 0};
public static int xLength;
public static int yLength;
public int getMaximumGold(int[][] grid) {
ans = 0;
xLength = grid.length;
yLength = grid[0].length;
isVisited = new boolean[xLength][yLength];
for (int i = 0; i < xLength; ++i){
for (int j = 0; j < yLength; ++j){
if (grid[i][j] != 0){
deepFirstSearch(i, j, grid, grid[i][j]);
}
}
}
return ans;
}
public static void deepFirstSearch(int x, int y, int[][] grid, int value){
isVisited[x][y] = true;
if (reachBottom(x, y, grid)){
ans = Math.max(ans, value);
return;
}
for (int i = 0; i < 4; ++i){
int newX = x + dX[i];
int newY = y + dY[i];
if (isValid(newX, newY) && !isVisited[newX][newY] && grid[newX][newY] != 0){
deepFirstSearch(newX, newY, grid, value + grid[newX][newY]);
isVisited[newX][newY] = false;//让所有的路径都能够被遍历到
}
}
//遍历所有路径(同一个格子能够被访问多次)
isVisited[x][y] = false;
}
public static boolean isValid(int x, int y){
return x >= 0 && x < xLength && y >= 0 && y < yLength;
}
//是否到底了
public static boolean reachBottom(int x, int y, int[][] grid){
for (int i = 0; i < 4; ++i){
if (isValid(x + dX[i], y + dY[i]) && grid[x + dX[i]][y + dY[i]] != 0){
if (!isVisited[x + dX[i]][y + dY[i]]) return false;
}
}
return true;
}
}
给定一个仅包含数字2-9
的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。
给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。
使用DFS
遍历所有情况,current
变量代表着当前的递归层次,当递归层次和数字长度一样时,返回结果。
class Solution {
public static char[][] numberOfCharacters;
public static StringBuilder sb;
public static ArrayList<String> res;
public List<String> letterCombinations(String digits) {
if (digits.length() == 0) return new ArrayList<>();
res = new ArrayList<>();
numberOfCharacters = new char[10][];
numberOfCharacters[2] = new char[]{'a', 'b', 'c'};
numberOfCharacters[3] = new char[]{'d', 'e', 'f'};
numberOfCharacters[4] = new char[]{'g', 'h', 'i'};
numberOfCharacters[5] = new char[]{'j', 'k', 'l'};
numberOfCharacters[6] = new char[]{'m', 'n', 'o'};
numberOfCharacters[7] = new char[]{'p', 'q', 'r', 's'};
numberOfCharacters[8] = new char[]{'t', 'u', 'v'};
numberOfCharacters[9] = new char[]{'w', 'x', 'y', 'z'};
sb = new StringBuilder();
deepFirstSearch(0, numberOfCharacters, digits);
return res;
}
public static void deepFirstSearch(int current ,char[][] chars, String digits){
if (current == digits.length()){
res.add(sb.toString());
return;
}
for (char i : chars[digits.charAt(current) - '0']){
sb.append(i);
deepFirstSearch(current + 1, chars, digits);
sb.deleteCharAt(sb.length() - 1);//进行回溯
}
}
}
数字 n
代表生成括号的对数,请你设计一个函数,用于能够生成所有可能的并且 有效的 括号组合。
朴素的方法是使用DFS
直接遍历长为2*n
的括号序列,同时判断括号序列是否合法,如果合法将其加入到答案中去。这个过程可以进行剪枝优化,具体的方法是,如果当前待加入答案的序列的左括号数量比右括号数量少,这表明这样的括号序列一定不合法。在遍历的过程中,一直维护好left
和right
变量,当right
和left
相等,同时递归层次达到了2*n
,就将结果加入到答案中去。
class Solution {
public static ArrayList<String> res;
public static StringBuilder sb;
public static int N;
public static int left = 0;
public static int right = 0;
public List<String> generateParenthesis(int n) {
N = n;
sb = new StringBuilder();
res = new ArrayList<>();
deepFirstSearch(0, new char[]{'(', ')'});
return res;
}
public static void deepFirstSearch(int current, char[] chars){
if (current == 2 * N){
if (left == right){
res.add(sb.toString());
}
return;
}
for (char i : chars){
sb.append(i);
if (i == '(') ++left;
else ++right;
if (left < right){ //不合法 剪枝回溯
sb.deleteCharAt(sb.length() - 1);
right--;
continue;
}
deepFirstSearch(current + 1, chars);
if (i == '(') --left;
else --right;
sb.deleteCharAt(sb.length() - 1);
}
}
}
给定一个 m x n
二维字符网格board
和一个字符串单词 word
。如果 word
存在于网格中,返回 true
;否则,返回false
。
单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不允许被重复使用。
使用DFS
搜索整个网格,同时进行贪心的选择,只有当下一个要选择的字符和word
的下一个字符相同,才进入下一层递归,否则就说明当前层的递归找不到答案,直接返回。
class Solution {
public static int[] dx = {0, 0, 1, -1};
public static int[] dy = {1, -1, 0, 0};
public static boolean[][] isVisited;
public static boolean res;
public static int xLength;
public static int yLength;
public boolean exist(char[][] board, String word) {
res = false;
xLength = board.length;
yLength = board[0].length;
if (xLength == 1 && yLength == 1 && word.length() == 1) return board[0][0] == word.charAt(0);
isVisited = new boolean[board.length][board[0].length];
for (int i = 0; i < xLength; ++i){
for (int j = 0; j < yLength; ++j){
if (board[i][j] == word.charAt(0)) deepFirstSearch(0, i, j, board, word);
}
}
return res;
}
public static void deepFirstSearch(int depth, int x, int y, char[][] board, String word){
isVisited[x][y] = true;
if (depth == word.length() - 1) {//如果这样的序列长度达到了要求 说明已经找到答案
res = true;
return;
}
for (int i = 0; i < 4; ++i){
int newX = x + dx[i];
int newY = y + dy[i];
if (newX >= 0 && newX < xLength && newY >= 0 && newY < yLength && !isVisited[newX][newY] && word.charAt(depth + 1) == board[newX][newY]){
deepFirstSearch(depth + 1, newX, newY, board, word);
isVisited[newX][newY] = false;//回溯
}
}
isVisited[x][y] = false;
}
}