代码随想录算法训练营29期Day28|LeetCode 93,78,90

文档讲解:复原IP地址 子集 子集II

93.复原IP地址

题目链接: https://leetcode.cn/problems/restore-ip-addresses/description/

思路:

这题其实不用搜索也能做,模拟即可。

题目中已经告诉我们,ip地址是由四个0-255之间的数字构成的。因此在给出的字符串中我们只需要枚举出三个隔板,也就是排列组合中的插板法。有三个隔板就能把原字符串分割成四部分,我们以此判断每部分是否有前导0,是否位于0-255之间就可以了。这样做时间复杂度肯定不会超,算一下组合数就知道了。

核心代码:

cpp 复制代码
class Solution {
private:
    vector<string> ans;
    void dfs(string s,int p,int q,int r){
        long long x=0;
        string ss="";
        for(int i=0;i<=p;i++) x*=10,x+=s[i]-'0',ss+=s[i];
        if((s[0]=='0'&&p>0)||x<0||x>255) return;
        ss+='.';x=0;
        for(int i=p+1;i<=q;i++) x*=10,x+=s[i]-'0',ss+=s[i];
        if((s[p+1]=='0'&&q>p+1)||x<0||x>255) return;
        ss+='.';x=0;
        for(int i=q+1;i<=r;i++) x*=10,x+=s[i]-'0',ss+=s[i];
        if((s[q+1]=='0'&&r>q+1)||x<0||x>255) return;
        ss+='.';x=0;
        for(int i=r+1;i<=s.length()-1;i++) x*=10,x+=s[i]-'0',ss+=s[i];
        if((s[r+1]=='0'&&s.length()-1>r+1)||x<0||x>255) return;
        ans.push_back(ss);
    }
public:
    vector<string> restoreIpAddresses(string s) {
        int n=s.length();
        if(n<4) return ans;
        for(int i=0;i<n-3;i++)
          for(int j=i+1;j<n-2;j++)
            for(int k=j+1;k<n-1;k++)
              dfs(s,i,j,k);
        return ans;
    }
};

78.子集

题目链接: https://leetcode.cn/problems/subsets/description/

思路:

本题很简单,就是给集合求所有子集嘛。那我们枚举集合中的每个数是否在集合中就可以了。利用dfs去做,递归边界条件为集合搜索到头,这时将枚举出的在子集中的数加入答案即可。时间复杂度为O(),n<=10,所以一定能过。

核心代码:

cpp 复制代码
class Solution {
private:
    vector<vector<int>> ans;
    vector<int> num;
    void dfs(vector<int> &nums,int x){
        if(x==nums.size()){
            ans.push_back(num);
            return;
        }
        dfs(nums,x+1);
        num.push_back(nums[x]);
        dfs(nums,x+1);
        num.pop_back();
        return;
    }
public:
    vector<vector<int>> subsets(vector<int>& nums) {
        dfs(nums,0);
        return ans;
    }
};

90.子集II

题目链接: https://leetcode.cn/problems/subsets-ii/description/

思路:

本题与LeetCode 40类似。我们可以采用类似的处理方法。

首先对原数据进行统计,统计出不同的数值以及每种数值出现的个数。

然后我们进行递归。在每层搜索中,我们枚举每种数值在子集中出现了多少个,从0依次至最大次数,然后将其在子集中加入对应次数。递归边界条件为每种数值都枚举了个数,将枚举出的子集加入答案,然后回溯即可。注意回溯时在子集中加的数要全部退掉。

核心代码:

cpp 复制代码
class Solution {
private:
    vector<pair<int,int>> tot;
    vector<vector<int>> ans;
    vector<int> num;
    void dfs(int x){
        if(x==tot.size()){
            ans.push_back(num);
            return;
        }
        for(int i=0;i<=tot[x].second;i++){
            dfs(x+1);
            num.push_back(tot[x].first);
        }
        for(int i=0;i<=tot[x].second;i++) num.pop_back();
        return;
    }
public:
    vector<vector<int>> subsetsWithDup(vector<int>& nums) {
        sort(nums.begin(),nums.end());
        for(int i:nums){
            if(tot.empty()||i!=tot.back().first) tot.emplace_back(i,1);
            else tot.back().second++;
        }
        dfs(0);
        return ans;
    }
};

今日总结

今日学习时长3h,题目还算可以。感冒差不多好了,状态很好,这几天要好好赶进度了。

相关推荐
乌萨奇也要立志学C++17 分钟前
【C++详解】STL-list模拟实现(深度剖析list迭代器,类模板未实例化取嵌套类型问题)
c++·list
yi.Ist32 分钟前
数据结构 —— 键值对 map
数据结构·算法
s1533535 分钟前
数据结构-顺序表-猜数字
数据结构·算法·leetcode
闻缺陷则喜何志丹36 分钟前
【前缀和 BFS 并集查找】P3127 [USACO15OPEN] Trapped in the Haybales G|省选-
数据结构·c++·前缀和·宽度优先·洛谷·并集查找
Coding小公仔38 分钟前
LeetCode 8. 字符串转换整数 (atoi)
算法·leetcode·职场和发展
GEEK零零七43 分钟前
Leetcode 393. UTF-8 编码验证
算法·leetcode·职场和发展·二进制运算
DoraBigHead2 小时前
小哆啦解题记——异位词界的社交网络
算法
序属秋秋秋2 小时前
《C++初阶之内存管理》【内存分布 + operator new/delete + 定位new】
开发语言·c++·笔记·学习
木头左3 小时前
逻辑回归的Python实现与优化
python·算法·逻辑回归
lifallen7 小时前
Paimon LSM Tree Compaction 策略
java·大数据·数据结构·数据库·算法·lsm-tree