Rust每日一练(Leetday0026) 最小覆盖子串、组合、子集

这篇具有很好参考价值的文章主要介绍了Rust每日一练(Leetday0026) 最小覆盖子串、组合、子集。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

Rust每日一练(Leetday0026) 最小覆盖子串、组合、子集

目录

76. 最小覆盖子串 Minimum Window Substring  🌟🌟🌟

77. 组合 Combinations  🌟🌟

78. 子集 Subsets  🌟🌟

🌟 每日一练刷题专栏 🌟

Rust每日一练 专栏

Golang每日一练 专栏

Python每日一练 专栏

C/C++每日一练 专栏

Java每日一练 专栏


76. 最小覆盖子串 Minimum Window Substring

给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 "" 。

注意:

  • 对于 t 中重复字符,我们寻找的子字符串中该字符数量必须不少于 t 中该字符数量。
  • 如果 s 中存在这样的子串,我们保证它是唯一的答案。

示例 1:

输入:s = "ADOBECODEBANC", t = "ABC"
输出:"BANC"

示例 2:

输入:s = "a", t = "a"
输出:"a"

示例 3:

输入: s = "a", t = "aa"
输出: ""
解释: t 中两个字符 'a' 均应包含在 s 的子串中,
因此没有符合条件的子字符串,返回空字符串。

提示:

  • 1 <= s.length, t.length <= 10^5
  • s 和 t 由英文字母组成

进阶:你能设计一个在 o(n) 时间内解决此问题的算法吗?

 代码1: 滑动窗口

use std::collections::HashMap;

fn min_window(s: String, t: String) -> String {
    let s: Vec<u8> = s.into_bytes();
    let t: Vec<u8> = t.into_bytes();
    if s.len() < t.len() {
        return String::new();
    }
    let mut need: HashMap<u8, i32> = HashMap::new(); // 存储t中每个字符的出现次数
    for i in 0..t.len() {
        *need.entry(t[i]).or_insert(0) += 1;
    }
    let (mut left, mut right) = (0, 0); // 滑动窗口的左右指针
    let mut count = t.len(); // 记录滑动窗口中还需要的字符数
    let mut min_len = s.len() + 1; // 记录最小覆盖子串的长度
    let mut start = 0; // 记录最小覆盖子串的起始位置
    while right < s.len() {
        // 当右指针指向的字符是需要的字符,count减一
        if let Some(v) = need.get_mut(&s[right]) {
            if *v > 0 {
                count -= 1;
            }
            *v -= 1;
        }
        right += 1;
        // 当count为0时,说明滑动窗口中已经包含t中的所有字符
        while count == 0 {
            // 如果当前的覆盖子串更小,则更新最小覆盖子串的长度和起始位置
            if right - left < min_len {
                min_len = right - left;
                start = left;
            }
            // 当左指针指向的字符是需要的字符,count加一
            if let Some(v) = need.get_mut(&s[left]) {
                *v += 1;
                if *v > 0 {
                    count += 1;
                }
            }
            left += 1;
        }
    }
    if min_len == s.len() + 1 {
        "".to_string()
    } else {
        String::from_utf8_lossy(&s[start..start + min_len]).to_string()
    }
}

fn main() {
    println!("{}", min_window("ADOBECODEBANC".to_string(), "ABC".to_string()));
    println!("{}", min_window("a".to_string(), "a".to_string()));
    println!("{}", min_window("a".to_string(), "aa".to_string()));
}

代码2: 双指针

use std::collections::HashMap;

fn min_window(s: String, t: String) -> String {
    let mut need: HashMap<char, i32> = HashMap::new();
    let mut count = t.len() as i32;
    let (mut left, mut right, mut start, mut min_len) = (0, 0, 0, s.len() + 1);

    for c in t.chars() {
        *need.entry(c).or_insert(0) += 1;
    }

    let s = s.chars().collect::<Vec<_>>();

    while right < s.len() {
        if let Some(v) = need.get_mut(&s[right]) {
            if *v > 0 {
                count -= 1;
            }
            *v -= 1;
        }
        right += 1;

        while count == 0 {
            if right - left < min_len {
                min_len = right - left;
                start = left;
            }
            if let Some(v) = need.get_mut(&s[left]) {
                *v += 1;
                if *v > 0 {
                    count += 1;
                }
            }
            left += 1;
        }
    }

    if min_len == s.len() + 1 {
        "".to_owned() // 返回空字符串
    } else {
        s[start..start + min_len].iter().collect()
    }
}

fn main() {
    println!("{}", min_window("ADOBECODEBANC".to_string(), "ABC".to_string()));
    println!("{}", min_window("a".to_string(), "a".to_string()));
    println!("{}", min_window("a".to_string(), "aa".to_string()));
}

输出:

BANC
a
//空行


77. 组合 Combinations

给定两个整数 n 和 k,返回范围 [1, n] 中所有可能的 k 个数的组合。

你可以按 任何顺序 返回答案。

示例 1:

输入:n = 4, k = 2
输出:
[
  [2,4],
  [3,4],
  [2,3],
  [1,2],
  [1,3],
  [1,4],
]

示例 2:

输入:n = 1, k = 1
输出:[[1]]

提示:

  • 1 <= n <= 20
  • 1 <= k <= n

代码1: 回溯法

fn combine(n: i32, k: i32) -> Vec<Vec<i32>> {
    let mut res: Vec<Vec<i32>> = vec![]; // 存储所有组合
    let mut path: Vec<i32> = vec![]; // 存储当前组合

    fn backtrack(start: i32, n: i32, k: i32, path: &mut Vec<i32>, res: &mut Vec<Vec<i32>>) {
        if path.len() == k as usize { // 当前组合长度为k,加入结果中
            res.push(path.clone());
            return;
        }
        for i in start..=n { // 枚举可选数字
            path.push(i); // 加入当前数字
            backtrack(i + 1, n, k, path, res); // 从i+1开始枚举下一个数字
            path.pop(); // 撤销当前数字
        }
    }

    backtrack(1, n, k, &mut path, &mut res); // 从1开始枚举第一个数字
    res
}

fn main() {
    println!("{:?}", combine(4, 2));
    println!("{:?}", combine(1, 1));
}

输出:

[[1, 2], [1, 3], [1, 4], [2, 3], [2, 4], [3, 4]]
[[1]]

代码2: 枚举法

fn combine(n: i32, k: i32) -> Vec<Vec<i32>> {
    let mut res: Vec<Vec<i32>> = vec![]; // 存储所有组合
    for i in 0..1<<n { // 枚举所有二进制数
        let mut path: Vec<i32> = vec![]; // 存储当前组合
        for j in 1..=n { // 枚举n个数字
            if i & 1 << j-1 != 0 { // 当前数字被选中
                path.push(j);
            }
        }
        if path.len() == k as usize { // 当前组合长度为k,加入结果中
            res.push(path);
        }
    }
    res
}

fn main() {
    println!("{:?}", combine(4, 2));
    println!("{:?}", combine(1, 1));
}

78. 子集 Subsets

给你一个整数数组 nums ,数组中的元素 互不相同 。返回该数组所有可能的子集(幂集)。

解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。

示例 1:

输入:nums = [1,2,3]
输出:[[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]]

示例 2:

输入:nums = [0]
输出:[[],[0]]

提示:

  • 1 <= nums.length <= 10
  • -10 <= nums[i] <= 10
  • nums 中的所有元素 互不相同

代码1:回溯法

fn subsets(nums: Vec<i32>) -> Vec<Vec<i32>> {
    let mut res: Vec<Vec<i32>> = vec![]; // 存储所有子集
    fn back_track (index: usize, path: &mut Vec<i32>, nums: &[i32], res: &mut Vec<Vec<i32>>) { // 递归枚举所有子集
        let tmp: Vec<i32> = path.to_vec(); // 将当前组合复制到临时数组中
        let n = nums.len();
        res.push(tmp); // 加入当前子集
        for i in index..n { // 枚举每个数字
            path.push(nums[i]); // 加入数字构成新的组合
            back_track (i+1, path, nums, res); // 递归枚举下一位数字
            path.pop(); // 移除数字
        }
    }
    back_track (0, &mut vec![], &nums, &mut res); // 从空集开始递归枚举所有子集
    res
}

fn main() {
    let nums: Vec<i32> = vec![1, 2, 3];
    println!("{:?}", subsets(nums));

    let nums: Vec<i32> = vec![0];
    println!("{:?}", subsets(nums));
}

输出:

[[], [1], [2], [1, 2], [3], [1, 3], [2, 3], [1, 2, 3]]
[[], [0]]

代码2: 循环枚举

fn subsets(nums: Vec<i32>) -> Vec<Vec<i32>> {
    let mut res: Vec<Vec<i32>> = vec![vec![]]; // 初始为空集
    for i in 0..nums.len() { // 枚举每个数字
        for sub in res.clone() { // 枚举已有的子集
            let mut temp = sub.clone();
            temp.push(nums[i]); // 加入当前数字
            res.push(temp); // 加入新的子集
        }
    }
    res
}

fn main() {
    let nums: Vec<i32> = vec![1, 2, 3];
    println!("{:?}", subsets(nums));

    let nums: Vec<i32> = vec![0];
    println!("{:?}", subsets(nums));
}

输出:

[[], [1], [2], [1, 2], [3], [1, 3], [2, 3], [1, 2, 3]]
[[], [0]]

代码3: 位运算

fn subsets(nums: Vec<i32>) -> Vec<Vec<i32>> {
    let mut res: Vec<Vec<i32>> = vec![]; // 存储所有子集
    let n = nums.len();
    for i in 0..(1 << n) { // 枚举所有二进制数
        let mut path: Vec<i32> = vec![]; // 存储当前子集
        for j in 0..n { // 枚举n个数字
            if i & (1 << j) != 0 { // 当前数字被选中
                path.push(nums[j]);
            }
        }
        res.push(path); // 加入当前子集
    }
    res
}

fn main() {
    let nums: Vec<i32> = vec![1, 2, 3];
    println!("{:?}", subsets(nums));

    let nums: Vec<i32> = vec![0];
    println!("{:?}", subsets(nums));
}

输出:

[[], [1], [2], [1, 2], [3], [1, 3], [2, 3], [1, 2, 3]]
[[], [0]]


🌟 每日一练刷题专栏 🌟

持续,努力奋斗做强刷题搬运工!

👍 点赞,你的认可是我坚持的动力! 

🌟 收藏,你的青睐是我努力的方向! 

评论,你的意见是我进步的财富!  

 主页:https://hannyang.blog.csdn.net/

Rust每日一练(Leetday0026) 最小覆盖子串、组合、子集

Rust每日一练 专栏

(2023.5.16~)更新中...

Rust每日一练(Leetday0026) 最小覆盖子串、组合、子集

Golang每日一练 专栏

(2023.3.11~)更新中...

Rust每日一练(Leetday0026) 最小覆盖子串、组合、子集

Python每日一练 专栏

(2023.2.18~2023.5.18)暂停更

Rust每日一练(Leetday0026) 最小覆盖子串、组合、子集

C/C++每日一练 专栏

(2023.2.18~2023.5.18)暂停更

Rust每日一练(Leetday0026) 最小覆盖子串、组合、子集

Java每日一练 专栏

(2023.3.11~2023.5.18)暂停更文章来源地址https://www.toymoban.com/news/detail-486519.html

到了这里,关于Rust每日一练(Leetday0026) 最小覆盖子串、组合、子集的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Rust每日一练(Leetday0025) 矩阵置零、搜索二维矩阵、颜色分类

    目录 73. 矩阵置零 Set Matrix Zeroes  🌟🌟 74. 搜索二维矩阵 Search A 2d-Matrix  🌟🌟 75. 颜色分类 Sort Colors  🌟🌟 🌟 每日一练刷题专栏 🌟 Rust每日一练 专栏 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 给定一个  m  x  n  的矩阵,如果一个元

    2024年02月11日
    浏览(28)
  • Rust每日一练(Leetday0012) 首末位置、插入位置、有效数独

    目录 34. 查找元素的首末位置 Find-first-and-last-position-of-element-in-sorted-array  🌟🌟 35. 搜索插入位置 Search Insert Position  🌟 36. 有效的数独 Valid Sudoku  🌟🌟 🌟 每日一练刷题专栏 🌟 Rust每日一练 专栏 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专

    2024年02月09日
    浏览(33)
  • Rust每日一练(Leetday0020) 最后单词的长度、螺旋矩阵II、排列序列

    目录 58. 最后一个单词的长度 Length of Last Word  🌟 59. 螺旋矩阵 II Spiral Matrix II  🌟🌟 60. 排列序列 Permutation Sequence  🌟🌟🌟 🌟 每日一练刷题专栏 🌟 Rust每日一练 专栏 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 给你一个字符串  s ,由

    2024年02月10日
    浏览(39)
  • Rust每日一练(Leetday0027) 单词搜索、删除重复项II、搜索旋转排序数组II

    目录 79. 单词搜索 Word Search  🌟🌟 80. 删除有序数组中的重复项 II Remove-duplicates-from-sorted-array-II  🌟🌟 81. 搜索旋转排序数组 II Search-in-rotated-sorted-array-II  🌟🌟 🌟 每日一练刷题专栏 🌟 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 给定一

    2024年02月08日
    浏览(33)
  • Golang每日一练(leetDay0052) 寻找旋转排序数组中的最小值I\II

    目录 153. 寻找旋转排序数组中的最小值 Find Minimum In Rotated Sorted Array  🌟🌟 154. 寻找旋转排序数组中的最小值 II Find Minimum In Rotated Sorted Array II  🌟🌟🌟 🌟 每日一练刷题专栏 🌟 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 已知一个长度为

    2024年02月16日
    浏览(31)
  • Java每日一练(20230516) 最小栈、组合总和II、相同的树

    目录 1. 最小栈  🌟 2. 组合总和 II  🌟🌟 3. 相同的树  🌟 🌟 每日一练刷题专栏 🌟 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 设计一个支持  push  , pop  , top  操作,并能在常数时间内检索到最小元素的栈。 push(x)  —— 将元素 x

    2024年02月05日
    浏览(26)
  • Golang每日一练(leetDay0031)

    目录 91. 解码方法  Decode Ways  🌟🌟 93. 复原 IP 地址 Restore IP Addresses  🌟🌟 🌟 每日一练刷题专栏 🌟 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 注:92.题 移到206.题之后 92. 反转链表 II Reverse Linked List II 一条包含字母  A-Z  的消息通过以

    2023年04月19日
    浏览(59)
  • Golang每日一练(leetDay0022)

    目录 64. 最小路径和 Minimum Path Sum  🌟🌟 65. 有效数字 Valid Number  🌟🌟🌟 66. 加一 Plus One  🌟 🌟 每日一练刷题专栏 🌟 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 给定一个包含非负整数的  m  x  n  网格  grid  ,请找出一条从左上角到

    2023年04月21日
    浏览(33)
  • Golang每日一练(leetDay0052)

    目录 153. 寻找旋转排序数组中的最小值 Find Minimum In Rotated Sorted Array  🌟🌟 154. 寻找旋转排序数组中的最小值 II Find Minimum In Rotated Sorted Array II  🌟🌟🌟 🌟 每日一练刷题专栏 🌟 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 已知一个长度为

    2024年02月02日
    浏览(31)
  • Golang每日一练(leetDay0004)

    目录 10. 正则表达式匹配 Regular Expression Matching  🌟🌟🌟 11. 盛最多水的容器 Container with most water  🌟🌟 12. 整数转罗马数字 Integer to Roman  🌟🌟 🌟 每日一练刷题专栏 🌟 Golang每日一练 专栏 Python每日一练 专栏 C/C++每日一练 专栏 Java每日一练 专栏 给你一个字符串 

    2023年04月08日
    浏览(28)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包