算法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;    
    }
};
相关推荐
sz66cm2 小时前
Python基础 -- 使用Python实现ssh终端并实现数据处理与统计功能
开发语言·python·ssh
小灰灰爱代码2 小时前
C++——求3*3矩阵对角元素之和。
数据结构·c++·算法
老K(郭云开)2 小时前
allWebPlugin中间件自定义alert、confirm及prompt使用
c++·chrome·中间件·prompt·html5·edge浏览器
liangbm33 小时前
MATLAB系列02:MATLAB基础
开发语言·数据结构·笔记·matlab·教程·工程基础·高级绘图
ac-er88884 小时前
如何在Flask中实现国际化和本地化
后端·python·flask
Adolf_19934 小时前
Flask-WTF的使用
后端·python·flask
空城皆是旧梦4 小时前
python爬虫初体验(一)
爬虫·python
藓类少女4 小时前
正则表达式
数据库·python·mysql·正则表达式
福鸦4 小时前
详解c++:new和delete
开发语言·c++
深蓝海拓4 小时前
迭代器和生成器的学习笔记
笔记·python·学习