算法D27|回溯算法4| 93.复原IP地址 78.子集 90.子集II

93.复原IP地址

本期本来是很有难度的,不过 大家做完 分割回文串 之后,本题就容易很多了

题目链接/文章讲解:代码随想录

视频讲解:回溯算法如何分割字符串并判断是合法IP?| LeetCode:93.复原IP地址_哔哩哔哩_bilibili

Python:

python 复制代码
class Solution:
    def __init__(self):
        self.result = []
        self.path = []

    def isvalid(self, s, start, end):
        if start>end: return False
        if s[start]=="0" and start!=end: return False
        return 0<=int(s[start:end+1])<=255

    def backtracking(self, s, start_index):
        if len(self.path)==4 and start_index==len(s):
            addr = ".".join(self.path)
            self.result.append(addr)
            return
        
        if len(self.path) > 4: return
                
        for i in range(start_index, min(start_index+3, len(s))):
            if self.isvalid(s, start_index, i):
                self.path.append(s[start_index:i+1])               
                self.backtracking(s, i+1)
                self.path.pop()
        return

    def restoreIpAddresses(self, s: str) -> List[str]:
        if len(s)<4 or len(s)>12: return []
        self.backtracking(s, 0)
        return self.result

C++:

C++版本写成直接在string里insert更简洁一些,C++没有类似python string.join的写法。

cpp 复制代码
class Solution {
public:
    vector<string> result;

    void backtracking(string& s, int startIndex, int pointNum) {
        if (pointNum == 3) {
            if (isValid(s, startIndex, s.size()-1)) {
                result.push_back(s);
            }
            return;   
        }
        for (int i = startIndex; i < s.size(); i++) {
            if (isValid(s, startIndex, i)) {
                s.insert(s.begin() + i + 1, '.');
                backtracking(s, i + 2, pointNum+1);
                s.erase(s.begin() + i + 1);
            } else break;
        }
    }

    bool isValid(const string&s, int start, int end) {
        if (start > end) return false;
        if (s[start] == '0' && start != end) return false;
        int num=0;
        for (int i=start; i<=end; i++) {
            if (s[i]>'9' || s[i]<'0') return false;
            num = num * 10 + (s[i] - '0');
            if (num>255) return false;
        }
        return true;
    }

    vector<string> restoreIpAddresses(string s) {
        result.clear();
        if (s.size()<4 || s.size()>12) return result;
        backtracking(s, 0, 0);
        return result;
    }
};

78.子集

子集问题,就是收集树形结构中,每一个节点的结果。 整体代码其实和 回溯模板都是差不多的。

题目链接/文章讲解:代码随想录

视频讲解:回溯算法解决子集问题,树上节点都是目标集和! | LeetCode:78.子集_哔哩哔哩_bilibili

本题比较简单

Python:

python 复制代码
class Solution:
    def __init__(self):
        self.result = []
        self.path = []
    
    def backtracking(self, nums, start_index):
        self.result.append(self.path[:])
        for i in range(start_index, len(nums)):
            self.path.append(nums[i])
            self.backtracking(nums, i+1)
            self.path.pop()
        return

    def subsets(self, nums: List[int]) -> List[List[int]]:
        self.backtracking(nums, 0)
        return self.result

C++:

cpp 复制代码
class Solution {
public:
    vector<vector<int>> result;
    vector<int> path;
    
    void backtracking(vector<int>& nums, int startIndex) {
        result.push_back(path);        // 要放在终止条件前面,否则回漏掉自己
        if (startIndex >= nums.size()) return;
        for (int i=startIndex; i<nums.size(); i++) {
            path.push_back(nums[i]);
            backtracking(nums, i+1);
            path.pop_back();
        }
        return;
    }

    vector<vector<int>> subsets(vector<int>& nums) {
        backtracking(nums, 0);
        return result;    
    }
};

90.子集II

大家之前做了 40.组合总和II 和 78.子集 ,本题就是这两道题目的结合,建议自己独立做一做,本题涉及的知识,之前都讲过,没有新内容。

题目链接/文章讲解:代码随想录

视频讲解:回溯算法解决子集问题,如何去重?| LeetCode:90.子集II_哔哩哔哩_bilibili

和上一题类似,区别在于去重,去重部分和40.组合总和II类似。

Python:

cpp 复制代码
class Solution:
    def __init__(self):
        self.result = []
        self.path = []

    def backtracking(self, nums, start_index):
        self.result.append(self.path[:])        
        for i in range(start_index, len(nums)):
            if i>start_index and nums[i]==nums[i-1]: continue # 去重
            self.path.append(nums[i])
            self.backtracking(nums, i+1)
            self.path.pop()
        return

    def subsetsWithDup(self, nums: List[int]) -> List[List[int]]:
        nums.sort()
        self.backtracking(nums, 0)
        return self.result

C++:

cpp 复制代码
class Solution {
public:
    vector<vector<int>> result;
    vector<int> path;
    
    void backtracking(vector<int>& nums, int startIndex) {
        result.push_back(path);
        for (int i=startIndex; i<nums.size(); i++) {
            if (i>startIndex && nums[i]==nums[i-1]) continue; //去重
            path.push_back(nums[i]);
            backtracking(nums, i+1);
            path.pop_back();
        }
        return;
    }

    vector<vector<int>> subsetsWithDup(vector<int>& nums) {
        sort(nums.begin(), nums.end());
        backtracking(nums, 0);
        return result;    
    }
};
相关推荐
载数而行5201 分钟前
算法集训1:模拟,枚举,错误分析,前缀和,差分
算法
weixin_408717777 分钟前
Go语言怎么编译Linux程序_Go语言编译Linux可执行文件教程【避坑】
jvm·数据库·python
hehelm18 分钟前
vector模拟实现
前端·javascript·算法
2501_9449347332 分钟前
财务岗位如何在工作中提升数据分析能力?从财务分析到经营分析这样练
python·信息可视化·数据分析
2301_8035545235 分钟前
C++ 并发核心:std::promise、std::future、std::async 超详细全解
开发语言·c++
2401_8654396339 分钟前
CSS怎么在flex布局中实现项目均分间距_设置justify-content space-evenly
jvm·数据库·python
m0_4939345340 分钟前
CSS如何实现输入框禁用样式_使用-disabled伪类设定
jvm·数据库·python
EverestVIP42 分钟前
C++ 成员函数的指针
c++
俺不要写代码1 小时前
线程启动、结束,创建线程多法、join,detach,线程的移动语义
服务器·开发语言·网络·c++
m0_716430071 小时前
如何用 classList.toggle 实现侧边栏菜单的点击展开与收起
jvm·数据库·python