算法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;    
    }
};
相关推荐
程序员编程指南28 分钟前
Qt 与 WebService 交互开发
c语言·开发语言·c++·qt·交互
荼蘼31 分钟前
基于 KNN 算法的手写数字识别项目实践
人工智能·算法·机器学习
赵英英俊33 分钟前
Python day26
开发语言·python
你怎么知道我是队长33 分钟前
python---eval函数
开发语言·javascript·python
溟洵36 分钟前
Qt 窗口 工具栏QToolBar、状态栏StatusBar
开发语言·前端·数据库·c++·后端·qt
Yuroo zhou1 小时前
IMU的精度对无人机姿态控制意味着什么?
单片机·嵌入式硬件·算法·无人机·嵌入式实时数据库
铭哥的编程日记1 小时前
《C++ list 完全指南:list的模拟实现》
c++
Rockson1 小时前
期货实时行情接口接入教程
python·api
程序员编程指南1 小时前
Qt 远程过程调用(RPC)实现方案
c语言·c++·qt·rpc·系统架构
jackzhuoa1 小时前
java小白闯关记第一天(两个数相加)
java·算法·蓝桥杯·期末