代码随想录算法训练营Day28 | 93.复原IP地址、78.子集、90.子集II | Python | 个人记录向

93.复原IP地址

代码随想录:93.复原IP地址
Leetcode:93.复原IP地址

做题

不知道回溯怎么写。

看文章

class Solution:
    def restoreIpAddresses(self, s: str) -> List[str]:
        result = []
        self.backtracking(s, 0, 0, "", result)
        return result

    def backtracking(self, s, start_index, point_num, current, result):
        if point_num == 3:  # 逗点数量为3时,分隔结束
            if self.is_valid(s, start_index, len(s) - 1):  # 判断第四段子字符串是否合法
                current += s[start_index:]  # 添加最后一段子字符串
                result.append(current)
            return

        for i in range(start_index, len(s)):
            if self.is_valid(s, start_index, i):  # 判断 [start_index, i] 这个区间的子串是否合法
                sub = s[start_index:i + 1]
                self.backtracking(s, i + 1, point_num + 1, current + sub + '.', result)
            else:
                break

    def is_valid(self, s, start, end):
        if start > end:
            return False
        if s[start] == '0' and start != end:  # 0开头的数字不合法
            return False
        num = 0
        for i in range(start, end + 1):
            if not s[i].isdigit():  # 遇到非数字字符不合法
                return False
            num = num * 10 + int(s[i])
            if num > 255:  # 如果大于255了不合法
                return False
        return True

时间复杂度: O(3^4),IP地址最多包含4个数字,每个数字最多有3种可能的分割方式,则搜索树的最大深度为4,每个节点最多有3个子节点。
空间复杂度: O(n)

另一种实现方法如下:(个人更喜欢这种)

class Solution:
    def restoreIpAddresses(self, s: str) -> List[str]:
        results = []
        self.backtracking(s, 0, [], results)
        return results

    def backtracking(self, s, index, path, results):
        if index == len(s) and len(path) == 4:
            results.append('.'.join(path))
            return

        if len(path) > 4:  # 剪枝
            return

        for i in range(index, min(index + 3, len(s))):
            if self.is_valid(s, index, i):
                sub = s[index:i+1]
                path.append(sub)
                self.backtracking(s, i+1, path, results)
                path.pop()

    def is_valid(self, s, start, end):
        if start > end:
            return False
        if s[start] == '0' and start != end:  # 0开头的数字不合法
            return False
        num = int(s[start:end+1])
        return 0 <= num <= 255

78.子集

代码随想录:78.子集
Leetcode:78.子集

做题

思路简单。

class Solution:
    def subsets(self, nums: List[int]) -> List[List[int]]:
        self.size = len(nums)
        self.res = []
        self.path = []
        self.backtracking(nums, 0)
        return self.res

    def backtracking(self, nums, start):
        self.res.append(self.path[:])

        for i in range(start, self.size):
            self.path.append(nums[i])
            self.backtracking(nums, i+1)
            self.path.pop()

时间复杂度: O(n * 2^n)
空间复杂度: O(n)

看文章

需要遍历整棵树,思路一样。

90.子集II

代码随想录:90.子集II
Leetcode:90.子集II

做题

思路没错,要注意:去重需要对数组进行排序

class Solution:
    def subsetsWithDup(self, nums: List[int]) -> List[List[int]]:
        self.size = len(nums)
        self.res = []
        self.path = []
        nums.sort()
        self.backtracking(nums, 0)
        return self.res

    def backtracking(self, nums, start):
        self.res.append(self.path[:])

        for i in range(start, self.size):
            if i > start and nums[i] == nums[i-1]:
                continue
            self.path.append(nums[i])
            self.backtracking(nums, i+1)
            self.path.pop()

时间复杂度: O(n * 2^n)
空间复杂度: O(n)

看文章

思路一致,也可以用used数组、set进行去重,但最好的还是上面的实现方法。

以往忽略的知识点小结

  • 对于特殊子集问题(比如判断ip),重点是在dfs时,取出这一条树上的数值(使用 start 和 i ),然后做判断(判断是否是ip)

个人体会

完成时间:2h。
心得:主要还是在处理特殊子集上比较有难度,常规的找子集和去重已经没什么问题。

  • 18
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值