【代码随想录算法训练营Day27】93.复原IP地址;78.子集;90.子集II

文章目录

❇️Day 27 第七章 回溯算法 part04

✴️今日内容

  • 93.复原IP地址
  • 78.子集
  • 90.子集II

❇️93.复原IP地址

自己的思路

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

自己的代码

不需要用path存储答案了,只需要在原数组s上插入'.'

java 复制代码
class Solution {
    List<String> res = new ArrayList<>();
    public List<String> restoreIpAddresses(String s) {
            //利用StringBuilder类型好操作
        StringBuilder sb = new StringBuilder(s);
        dfs(sb, 0, 0);
        return  res;
    }
    //dotCount: 记录添加逗点的数量
    public void dfs(StringBuilder s, int startIndex, int dotCount){
        if(dotCount == 3){
            if(isValid(s, startIndex, s.length() - 1)){
                res.add(s.toString());
            }
            return;
        }
        for (int i = startIndex; i < s.length(); i++) {
            //判断分割出来的子串是否合法
            if(isValid(s, startIndex, i)){
                s.insert(i + 1, '.');
                dfs(s, i + 2, dotCount + 1);
                s.deleteCharAt(i + 1);        //回溯
            }else break;
        }
        return;
    }
    //[start, end]
    private boolean isValid(StringBuilder s, int start, int end) {
        if (start > end) {
            return false;
        }
        if (s.charAt(start) == '0' && start != end) { // 0开头的数字不合法
            return false;
        }
        int num = 0;
        for (int i = start; i <= end; i++) {
            if (s.charAt(i) > '9' || s.charAt(i) < '0') { // 遇到⾮数字字符不合法
                return false;
            }
            num = num * 10 + (s.charAt(i) - '0');
            if (num > 255) { // 如果⼤于255了不合法
                return false;
            }
        }
        return true;
    }
}

❇️78.子集

(3.8补)有点忘了回溯法了,看了下文章思路

随想录思路

以示例中nums = [1,2,3]为例把求子集抽象为树型结构,如下:

从图中红线部分,可以看出遍历这个树的时候,把所有节点都记录下来,就是要求的子集集合。

自己的思路

类似组合

  1. 确定递归函数的参数和返回值
    1. nums
    2. startIndex:确定开始下标
  2. 确定终止条件
    1. 集合为空
  3. 确定单层递归的逻辑

自己的代码

严格按照模板来,和组合逻辑一样,只是添加逻辑不一样

java 复制代码
class Solution {
    List<List<Integer>> res = new ArrayList<>();
    LinkedList<Integer> path = new LinkedList<>();
    public List<List<Integer>> subsets(int[] nums) {
        dfs(nums, 0);
        return res;
    }
    //可以把dfs函数改成void
    public List<List<Integer>> dfs(int[] nums, int startIndex){
        res.add(new ArrayList<>(path));
        if(startIndex >= nums.length) return res;
        for (int i = startIndex; i < nums.length; i++) {
            path.push(nums[i]);
            dfs(nums, i + 1);
            path.pop();
        }
        return res;
    }
}

❇️90.子集II

自己的思路

和上一题一样只需要遇到相同数时判断一下前面相同的数是否被用过,所以添加参数boolean visited[]判断数有没有被用过

自己的代码

java 复制代码
class Solution {
    List<List<Integer>> res = new ArrayList<>();
    LinkedList<Integer> path = new LinkedList<>();
    boolean[] visited;
    public List<List<Integer>> subsetsWithDup(int[] nums) {
        if (nums.length == 0){
            res.add(path);
            return res;
        }
        Arrays.sort(nums);
        visited = new boolean[nums.length];
        dfs(nums, 0);
        return res;
    }
    public void dfs(int[] nums, int start){
        res.add(new ArrayList<>(path));
        if(start >= nums.length) {
            return;
        }
        for (int i = start; i < nums.length; i++) {
            if(i > 0 && nums[i] == nums[i - 1] && !visited[i - 1]){
                continue;
            }
            visited[i] = true;
            path.push(nums[i]);
            dfs(nums, i + 1);
            path.pop();
            visited[i] = false;
        }
    }
}
相关推荐
GEEK零零七1 小时前
Leetcode 3299. 连续子序列的和
算法·leetcode·动态规划
飞飞是甜咖啡1 小时前
【机器学习】Teacher-Student框架
人工智能·算法·机器学习
蒟蒻小袁2 小时前
力扣面试150题--单词接龙
算法·leetcode·面试
ghie90902 小时前
LMD分解通过局部均值分解重构信号实现对信号的降噪
算法·均值算法·重构
零叹3 小时前
篇章十 数据结构——排序
java·数据结构·算法·排序算法
涛哥码咖3 小时前
前端十种排序算法解析
前端·算法·排序算法
朝朝又沐沐4 小时前
算法竞赛阶段二-数据结构(32)数据结构简单介绍
数据结构·算法
共享家95274 小时前
c语言(重点)
c语言·数据结构·算法
玉米的玉*」*4 小时前
【每日likou】704. 二分查找 27. 移除元素 977.有序数组的平方
数据结构·算法·leetcode
星火飞码iFlyCode4 小时前
【无标题】
java·前端·人工智能·算法