[LeetCode 1871] 跳跃游戏 7(Ⅶ)

题面:

数据范围:

2 ≤ s . l e n g t h ≤ 1 0 5 2 \le s.length \le 10^5 2≤s.length≤105
s i si si 要么是 ′ 0 ′ '0' ′0′ ,要么是 ′ 1 ′ '1' ′1′
s 0 = = 0 s0 == 0 s0==0
1 ≤ m i n J u m p ≤ m a x J u m p < s . l e n g t h 1 \le minJump \le maxJump < s.length 1≤minJump≤maxJump<s.length

思路 & Code

重点: 当 s i = = 0 si==0 si==0 时,才能继续跳,也才能被跳到。

So,自然会想到存储所有能够跳到的 0 0 0 点,而当前 0 0 0 点必然是从存储的 0 0 0 点跳过来的。容易想到可以暴力状态转移

用一数组 d p dp dp 存储下标状态:

  1. d p i n d e x = = 0 dpindex==0 dpindex==0,说明 i n d e x index index 位置没有被跳到
  2. d p i n d e x = = 1 dpindex==1 dpindex==1, i n d e x index index 可以被跳到

那么,对于当前点 i n d e x index index 都得判断 m i n ( i − m a x J u m p ) , 0 ) , i − m i n J u m p min(i-maxJump), 0), i-minJump min(i−maxJump),0),i−minJump 区间内的所有可跳点,这样太暴力了, O ( n k ) O(nk) O(nk)。必须考虑优化。

差分维护

时间复杂度: O ( n ) O(n) O(n)
空间复杂度: O ( n ) O(n) O(n)

维护个前缀和,每次到 s i = = ′ 0 ′ si=='0' si==′0′(可跳点)时,判断区间 m i n ( i − m a x J u m p ) , 0 ) , i − m i n J u m p min(i-maxJump), 0), i-minJump min(i−maxJump),0),i−minJump是否存在可跳点 0 0 0,即 p r e i − m i n J u m p − p r e i − m a x J u m p − 1 prei-minJump - prei-maxJump-1 prei−minJump−prei−maxJump−1是否大于零。(当然了,区间得注意下)

Code:

cpp 复制代码
bool canReach(string s, int minJump, int maxJump) {
    int n = s.size();
    vector<int> dp(n, 0), pre(n, 0);
    dp[0] = 1;
    for(int i = 0; i < minJump; ++i) pre[i] = 1;
    for(int i = minJump; i < n; ++i) {
        int l = i - maxJump, r = i - minJump;
        // 差分判断可跳区间内是否存在 0 点
        if(s[i] == '0' && pre[r] - (l >= 1 ? pre[l - 1] : 0) > 0) dp[i]=1;
        pre[i] = pre[i-1] + dp[i];
    }
    return dp[n-1];
    /* 
    当然了,不用dp数组维护也可以
    写成:
    for(int i = minJump; i < n; ++i) {
		int l = i - maxJump, r = i - minJump;
		if(s[i] == '0' && pre[r] - (l >= 1 ? pre[l - 1] : 0) > 0) {
			if(i == n-1) return true;
			pre[i] = pre[i-1]+1;
		} else pre[i] = pre[i-1];
	}
	return false;
    */
}

当然了,由于可跳区间其实就是一个滑动窗口,直接拿个临时变量存储可跳点个数就行,但得存个状态哈

Code:

cpp 复制代码
bool canReach(string s, int minJump, int maxJump) {
    int n = s.size(), cnt = 1;
    vector<bool> dp(n, false);
    dp[0] = true;
    for(int i = minJump; i < n; ++ i) {
        if(s[i] == '0' && cnt) dp[i] = true;
        // 可跳区间要右移啦(其实就是动态维护滑动窗口的思想啦)
        // 如果左端点是可跳的,cnt--;如果要加入的点(右端点)是可跳的,cnt++
        if(i - maxJump >= 0 && dp[i - maxJump]) --cnt;
        if(i - minJump + 1 < n && dp[i - minJump + 1]) ++ cnt;
    }
    return dp[n-1];
}

动态维护滑动窗口(可跳区间)

上面都提到滑动窗口了,那直接拿个队列去维护滑动窗口也很简单啦。时间复杂度和空间复杂度都是 O ( n ) O(n) O(n)。

Code:

cpp 复制代码
bool canReach(string s, int minJump, int maxJump) {
    int n = s.size();
    if(s[n-1] == '1') return false;
    queue<int> q;
    q.push(0);
    for(int i = minJump; i < n; ++i) {
        if(s[i] == '1') continue;
        while(!q.empty() && q.front() < i - maxJump) q.pop();
        if(!q.empty() && q.front() <= i - minJump) {
            if(i == n-1) return true;
            q.push(i);
        }
    }
    return false;
}
相关推荐
8Qi832 分钟前
LeetCode 516:最长回文子序列
算法·leetcode·职场和发展·动态规划
youngerwang2 小时前
【从搬运工到协处理器:网卡芯片架构、算法、验证与边缘演进深度剖析】
网络·算法·架构·芯片
KaMeidebaby2 小时前
卡梅德生物技术快报|纯化重组蛋白实操详解
人工智能·python·tcp/ip·算法·机器学习
手写码匠3 小时前
从零实现 Prompt 工程引擎:结构化提示、自动优化与多轮自省体系
人工智能·深度学习·算法·aigc
无限码力3 小时前
阿里算法岗 0530笔试真题 - 多约束条件下的元素匹配统计
算法·阿里笔试真题·阿里机试真题·阿里算法岗笔试
lqqjuly3 小时前
MLA — 多头潜在注意力深度解析
深度学习·神经网络·算法
吴可可1234 小时前
SolidWorks草图转三维DWG技巧
算法
redaijufeng4 小时前
C++雾中风景7:闭包
c++·算法·风景
小欣加油5 小时前
leetcode287寻找重复数
数据结构·c++·算法·leetcode
尽兴-5 小时前
2.1 向量基础:Embedding、余弦相似度、欧氏距离、向量检索
算法·embedding·欧氏距离·向量检索·余弦相似度