天天看点

数据结构与算法-递归-leetcode练习

递归

编写递归要决,千万不要用人脑去递和归每一层,正确的做法是找到如何分解为子问题,然后几个子问题的关系是什么。然后翻译成代码。

leetcode17

题目描述:

给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。

给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。

数据结构与算法-递归-leetcode练习

示例:

输入:"23"

输出:["ad", "ae", "af", "bd", "be", "bf", "cd", "ce", "cf"].

说明:

尽管上面的答案是按字典序排列的,但是你可以任意选择答案输出的顺序。

来源:力扣(LeetCode)

链接:https://leetcode-cn.com/problems/letter-combinations-of-a-phone-number

著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解题思路

先说最简单的,对应数字与字符的对应关系,我们可以用数组来记录,下标为代表0到9的数字,值即为每个字符串,由于0和1都没有字符,为了使用方便,还是使用空字符来占用。即可表示为

String[] dict = {"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};

再来说如何组合所有可能性,还不能重复。比如输入23

类比我们人脑来处理的过程就是这样:

先取2的第一个字符与后面每个字符组合,然后再取出3的每个字符与后面组合即可得到所有可能性。

使用递归还求解,需定能定义出来递推公式,可我数学公式基本已经还给老师了,实在是对不起老师呀,还是来想想处理过程吧。

递归需要我们将问题分解为多个子问题。那我们如何分解呢?

刚刚人脑的处理过程就取出数字对应的字符串,然后一个一个的遍历,与之前的字符进行组合,这是递归的处理逻辑,当我们遍历到输入数字最后一个时,就需要将每个数字组合进行记录。这是递归的终止条件。

然后我们此翻译成代码:

解题实现

public class RecurstionSolution {

  private static final String[] STAT_CODE =

      new String[] {"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};

  public List<String> letterCombinations(String digits) {

    List<String> result = new ArrayList<>();

    if (digits == null || digits.isEmpty()) {

      return result;

    }

    String inputData = digits.trim();

    // 1,将输入的字符转化为数字

    int[] dataNum = new int[inputData.length()];

    for (int i = 0; i < inputData.length(); i++) {

      dataNum[i] = inputData.charAt(i) - '0';

    }

    String dataChar = "";

    this.recurstionData(dataNum, 0, result, dataChar);

    return result;

  }

  private void recurstionData(int[] dataNum, int index, List<String> list, String dataChar) {

    // 1,递归的终止条件

    if (index == dataNum.length) {

      list.add(new String(dataChar));

      return;

    }

    // 进行递归的求解

    for (int i = 0; i < STAT_CODE[dataNum[index]].length(); i++) {

      dataChar = dataChar + STAT_CODE[dataNum[index]].charAt(i);

      // 当取出一个字符后继续进行遍历

      this.recurstionData(dataNum, index + 1, list, dataChar);

      // 当遍历完成后,需要删除当前的最后一个字符,以可以与其他进行组合

      dataChar = dataChar.substring(0, dataChar.length() - 1);

    }

  }

}

leetcod22

题目描述:

22. 括号生成

给出 n 代表生成括号的对数,请你写出一个函数,

使其能够生成所有可能的并且有效的括号组合。

例如,给出 n = 3,生成结果为:

[

  "((()))",

  "(()())",

  "(())()",

  "()(())",

  "()()()"

]

来源:力扣(LeetCode)

链接:https://leetcode-cn.com/problems/generate-parentheses

著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解题思路:

这个题目说实话我也做错了,我开始的解题思路是以成对的括号为单位,我想的是在括号的左边 ,右边,还有中间,依次的插入号位,递归这样就可以求解了,但当我实现的时候,我发现,我的思路是错误的,因为当遍历到多层时,这个中间括号已经无法再明确的定义,需要寻找其他解法。

我又来参照了其他的方案,终于理解了这个递归的求解方法:

这个问题的求解可以分为两部分,添加左括号、和添加右括号,添加的个数正好是输入的数字,例如3,就是可以添加3个左括号和3个右括号。

那现在就是如何添加左括号和右括号的问题:

先说左括号,最大左括号不就是输入的数字么,所以需要一个左括号的索引来记录当前是第几个左括号,需要小于最大左括号。添加一个括号即可

当左括号添加了,那右括号嗯,是否和左括号一样,这个还是有点不一样呢,右括号需要与左括号成对出现,就不能出现右括号比左括号还多的情况,这样最大右括号需要小于当前的左括号数了。

递归终止条件是什么呢,当然是当前括号数满足了输入的括号数*2,例如输入3,就是6个,左路输入左右括号数,一样要是6个。现在来翻译成代码吧.

递归代码实现

public class SolutionParthese2 {

  public static final String PARENTHESE_LEFT = "(";

  public static final String PARENTHESE_RIGHT = ")";

  public List<String> generateParenthesis(int num) {

    List<String> result = new ArrayList<>();

    if (num == 1) {

      result.add(PARENTHESE_LEFT + PARENTHESE_RIGHT);

      return result;

    }

    if (num == 0) {

      return result;

    }

    this.recurstion(num, 0, 0, result, "");

    return result;

  }

  public void recurstion(int maxNum, int left, int right, List<String> resultLst, String data) {

    // 1,首先写递归的终止条件

    if (data.length() == maxNum * 2) {

      resultLst.add(data);

      return;

    }

    // 添加左括号

    if (left < maxNum) {

      this.recurstion(maxNum, left + 1, right, resultLst, data + PARENTHESE_LEFT);

    }

    // 添加右括号,右括号,需要小于左括号,否则不对成对了

    if (right < left) {

      this.recurstion(maxNum, left, right + 1, resultLst, data + PARENTHESE_RIGHT);

    }

  }

}

leetcode39

题目描述:

39. 组合总和

给定一个无重复元素的数组 candidates 和一个目标数 target ,

找出 candidates 中所有可以使数字和为 target 的组合。

candidates 中的数字可以无限制重复被选取。

说明:

所有数字(包括 target)都是正整数。

解集不能包含重复的组合。 

示例 1:

输入: candidates = [2,3,6,7], target = 7,

所求解集为:

[

  [7],

  [2,2,3]

]

示例 2:

输入: candidates = [2,3,5], target = 8,

所求解集为:

[

  [2,2,2,2],

  [2,3,3],

  [3,5]

]

来源:力扣(LeetCode)

链接:https://leetcode-cn.com/problems/combination-sum

著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解题思路:

这个是一个求组合的问题,我的求解思路是这样子的,将数组中的每个元素,从当前自己开始,依次与一个元素相加,如果超过了,说明符不合,如果与预期值相同,则记录下来,未到达退出条件则继续递归。为了确保无重复的数据,可使用set来进行存储

编码实现

public class SolutionMySelf {

  public List<List<Integer>> combinationSum(int[] inputSrc, int target) {

    if (inputSrc == null || inputSrc.length == 0) {

      return null;

    }

    List<Integer> currList = new ArrayList<>();

    Set<List<Integer>> collect = new HashSet<>();

    this.recurstion(inputSrc, target, 0, 0, currList, collect);

    List<List<Integer>> result = new ArrayList<>(collect);

    return result;

  }

  private void recurstion(

      int[] inputSrc,

      int target,

      int value,

      int startIndex,

      List<Integer> collect,

      Set<List<Integer>> resultList) {

    // 1,递归的终止条件

    if (value == target) {

      resultList.add(new ArrayList<>(collect));

      return;

    }

    // 2,超过出说明不符合直接退出

    if (value > target) {

      return;

    }

    for (int i = startIndex; i < inputSrc.length; i++) {

      value = value + inputSrc[i];

      collect.add(inputSrc[i]);

      this.recurstion(inputSrc, target, value, i, collect, resultList);

      // 当结束后,需要移除最后一个元素

      collect.remove(collect.size() - 1);

      value = value - inputSrc[i];

    }

  }

}

leetcode46

题目描述:

给定一个没有重复数字的序列,返回其所有可能的全排列。

示例:

输入: [1,2,3]

输出:

[

  [1,2,3],

  [1,3,2],

  [2,1,3],

  [2,3,1],

  [3,1,2],

  [3,2,1]

]

来源:力扣(LeetCode)

链接:https://leetcode-cn.com/problems/permutations

著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解题思路:

我的解题思路就是将数组中的每个元素与剩下的两个元素进行组合,当给合的大小与数组大小一致就放入到结果集中。那如何表示剩余下元素呢,我的办法是将已经遍历过程下标通过一个boolean数组记录下来,这样就能知道了

代码实现

public class SolutionPermute {

  public List<List<Integer>> permute(int[] nums) {

    Set<List<Integer>> data = new HashSet<>();

    List<Integer> currdata = new ArrayList<>();

    boolean[] exists = new boolean[nums.length];

    this.recurstion(nums, data, currdata, exists);

    List<List<Integer>> result = new ArrayList<>(data);

    return result;

  }

  private void recurstion(

      int[] nums, Set<List<Integer>> data, List<Integer> currdata, boolean[] exists) {

    // 如果当前组合到达组合的结果大小,则加入到结果集中

    if (currdata.size() == nums.length) {

      data.add(new ArrayList<>(currdata));

      return;

    }

    for (int i = 0; i < nums.length; i++) {

      // 仅对未遍历过元素进行遍历

      if (!exists[i]) {

        exists[i] = true;

        currdata.add(nums[i]);

        this.recurstion(nums, data, currdata, exists);

        currdata.remove(currdata.size() - 1);

        exists[i] = false;

      }

    }

  }

}

leetcode79

题目描述:

79. 单词搜索

给定一个二维网格和一个单词,找出该单词是否存在于网格中。

单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不允许被重复使用。

示例:

board =

[

  ['A','B','C','E'],

  ['S','F','C','S'],

  ['A','D','E','E']

]

给定 word = "ABCCED", 返回 true.

给定 word = "SEE", 返回 true.

给定 word = "ABCB", 返回 false.

来源:力扣(LeetCode)

链接:https://leetcode-cn.com/problems/word-search

著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解题思路

针对此单词搜索,因为存在4种匹配的情况:

情况1,横坐标+1可匹配

情况2,横坐标-1可匹配

情况3,纵坐标+1可匹配

情况4,纵坐标-1可匹配

递归的终止条件呢?当发生字符不匹配,搜索终止,或者搜索到匹配字符,搜索终止,那过有其他条件吗?当然有,那就是横纵坐标,必须在字符的下标范围以内

代码实现

public class SolutionRectustionMySelf {

  public boolean exist(char[][] src, String data) {

    boolean[][] exists = new boolean[src.length][src[0].length];

    boolean existsResult = false;

    String srcvalue = null;

    //搜索开始的字符位置

    for (int i = 0; i < src.length; i++) {

      for (int j = 0; j < src[i].length; j++) {

        srcvalue = String.valueOf(src[i][j]);

        if (data.startsWith(srcvalue)) {

          existsResult = this.exist(src, "", data, i, j, exists);

          if (existsResult) {

            return true;

          }

        }

      }

    }

    return false;

  }

  public boolean exist(

      char[][] src,

      String currData,

      String targetData,

      int rowindex,

      int cellindex,

      boolean[][] exists) {

    if (currData.equals(targetData)) {

      return true;

    }

    if (rowindex < 0 || rowindex >= src.length) {

      return false;

    }

    if (cellindex < 0 || cellindex >= src[rowindex].length) {

      return false;

    }

    // 节点已经访问,则标识为false

    if (exists[rowindex][cellindex]) {

      return false;

    }

    currData = currData + src[rowindex][cellindex];

    if (currData.length() > targetData.length()) {

      return false;

    }

    // 在进入时标识已经访问

    exists[rowindex][cellindex] = true;

    // 当字符匹配的时,可能存在4个方向的情况

    if (targetData.startsWith(currData)) {

      if (exist(src, currData, targetData, rowindex + 1, cellindex, exists)

          || exist(src, currData, targetData, rowindex, cellindex + 1, exists)

          || exist(src, currData, targetData, rowindex - 1, cellindex, exists)

          || exist(src, currData, targetData, rowindex, cellindex - 1, exists)) {

        return true;

      }

    }

    // 退出时还需要标识出未访问

    exists[rowindex][cellindex] = false;

    return false;

  }

}