• 代码随想录算法训练营第28天|● 93.复原IP地址 ● 78.子集 ● 90.子集II


    一、93.复原IP地址

    力扣

    切割问题可以抽象为树型结构,如图:

    • 递归参数

    131.分割回文串 (opens new window)中我们就提到切割问题类似组合问题。

    startIndex一定是需要的,因为不能重复分割,记录下一层递归分割的起始位置。

    本题我们还需要一个变量pointNum,记录添加逗点的数量。

    • 递归终止条件

    终止条件和131.分割回文串 (opens new window)情况就不同了,本题明确要求只会分成4段,所以不能用切割线切到最后作为终止条件,而是分割的段数作为终止条件。

    pointNum表示逗点数量,pointNum为3说明字符串分成了4段了。

    然后验证一下第四段是否合法,如果合法就加入到结果集里

    1. class Solution {
    2. List<String> result = new ArrayList<>();
    3. public List<String> restoreIpAddresses(String s) {
    4. if (s.length() > 12) return result;
    5. backtracking(s, 0, 0);
    6. return result;
    7. }
    8. private void backtracking(String s, int startIndex, int pointNum) {
    9. if (pointNum == 3) {
    10. // 判断第四段⼦字符串是否合法,如果合法就放进result中
    11. if (isValid(s,startIndex,s.length()-1)) {
    12. result.add(s);
    13. }
    14. return;
    15. }
    16. for (int i = startIndex; i < s.length(); i++) {
    17. if (isValid(s, startIndex, i)) {
    18. s = s.substring(0, i + 1) + "." + s.substring(i + 1);
    19. pointNum++;
    20. backtracking(s, i + 2, pointNum);// 插⼊逗点之后下⼀个⼦串的起始位置为i+2
    21. pointNum--;// 回溯
    22. s = s.substring(0, i + 1) + s.substring(i + 2);// 回溯删掉逗点
    23. }else {
    24. break;
    25. }
    26. }
    27. }
    28. private boolean isValid(String s, int start, int end) {
    29. if (start > end) {
    30. return false;
    31. }
    32. if(s.charAt(start) == '0' && start != end) {// 0开头的数字不合法
    33. return false;
    34. }
    35. int num = 0;
    36. for (int i = start; i <= end; i++) {
    37. if (s.charAt(i) > '9' || s.charAt(i) < '0') {// 遇到⾮数字字符不合法
    38. return false;
    39. }
    40. num = num * 10 + (s.charAt(i) - '0');
    41. if (num > 255) { // 如果⼤于255了不合法
    42. return false;
    43. }
    44. }
    45. return true;
    46. }
    47. }

    二、78.子集

    力扣

    如果把 子集问题、组合问题、分割问题都抽象为一棵树的话,那么组合问题和分割问题都是收集树的叶子节点,而子集问题是找树的所有节点!

    其实子集也是一种组合问题,因为它的集合是无序的,子集{1,2} 和 子集{2,1}是一样的。

    那么既然是无序,取过的元素不会重复取,写回溯算法的时候,for就要从startIndex开始,而不是从0开始!

     

    剩余集合为空的时候,就是叶子节点。

    那么什么时候剩余集合为空呢?

    就是startIndex已经大于数组的长度了,就终止了,因为没有元素可取了,代码如下:

    1. if (startIndex >= nums.size()) {
    2. return;
    3. }
    • 单层搜索逻辑

    求取子集问题,不需要任何剪枝!因为子集就是要遍历整棵树

    1. class Solution {
    2. List<List<Integer>> result = new ArrayList<>();
    3. List<Integer> path = new ArrayList<>();
    4. public List<List<Integer>> subsets(int[] nums) {
    5. backTracking(nums, 0);
    6. return result;
    7. }
    8. public void backTracking(int[] nums, int startIndex) {
    9. result.add(path); //收集子集,要放在终止添加的上面,否则会漏掉自己
    10. if(startIndex > nums.length) {
    11. return;
    12. }
    13. for(int i = startIndex; i < nums.lenth; i++) {
    14. path.add(nums[i]);
    15. backTracking(nums, i + 1);
    16. path.remove(path.size() - 1);
    17. }
    18. }
    19. }

    三、90.子集II

    力扣

    回溯中去重,nums[i] == nums[i-1]。

             从图中可以看出,同一树层上重复取2 就要过滤掉,同一树枝上就可以重复取2,因为同一树枝上元素的集合才是唯一子集!

    1. class Solution {
    2. List<List<Integer>> result = new ArrayList<>();
    3. LinkedList<Integer> path = new LinkedList<>();
    4. public List<List<Integer>> subsetsWithDup(int[] nums) {
    5. Arrays.sort(nums);
    6. backTracking(nums, 0);
    7. return result;
    8. }
    9. public void backTracking(int[] nums, int startIndex) {
    10. result.add(new ArrayList(path));
    11. if(startIndex >= nums.length) {
    12. return;
    13. }
    14. for(int i = startIndex; i < nums.length; i++) {
    15. //去重
    16. if(i > startIndex && nums[i] == nums[i-1]) {
    17. continue;
    18. }
    19. path.add(nums[i]);
    20. backTracking(nums, i + 1);
    21. path.removeLast();
    22. }
    23. }
    24. }

  • 相关阅读:
    17【观察者设计模式】
    Sentinel-1主动微波数据下载
    UWA上新|真人真机测试新增海外机型专区
    为何抢跑了成长型企业SaaS赛道?先看看用友YonSuite有多了解企业
    03-jenkins集成环境配置
    ExtJS 数据处理-Validations(验证器)
    Vue插件的使用
    深入理解MySQL分区技术
    LIEF:修改安卓.so后报 dlopen failed:has invalid shdr offset/size
    再登GitHub下载榜首!阿里大牛分享内部漫谈程序员职场修炼手册
  • 原文地址:https://blog.csdn.net/weixin_44565532/article/details/127576808