牛客练习赛126(O(n)求取任意大小区间最值)

牛客练习赛126(O(n)求取任意大小区间最值)

牛客练习赛126

A.雾粉与签到题

题意:给出长度为n的数组, 顺序选出任意三个元素,最小化第二个元素

思路:

遍历除了第一个和最后一个元素取最小值即可

AC code:

c 复制代码
void solve() {
    int n; cin >> n;
    vector<int> a(n);
    map<int, int> mp;
    for (int i = 0; i < n; i ++) cin >> a[i];
    int mn = 1e18;
    for (int i = 1; i < n - 1; i ++) mn = min(mn, a[i]);
    cout << mn << endl;
}

B.雾粉与数论

题意:给你一个正整数 n n n,请你输出对每个 2 ≤ i ≤ n 2 \le i \le n 2≤i≤n, gcd ⁡ ( i ∗ ( i − 1 ) 2 , i ∗ ( i + 1 ) 2 ) \gcd(\frac {i*(i-1)} 2, \frac {i*(i+1)} 2) gcd(2i∗(i−1),2i∗(i+1)) 之和对 1 0 9 + 7 10^9+7 109+7 取模。

思路:打表发现规律,奇数直接对答案贡献自身,偶数贡献/2,注意,取模一定要最后取,过程中取模会影响最后结果。

AC code:

c 复制代码
void solve() {
    int n; cin >> n;
    int ans = (n * (n + 1) / 2 - 1);
    int ca = n / 2;
    int t = ca * (ca + 1) / 2;
    ans -= t;
    cout << ans % MOD << endl;
}

C.雾粉与最小值(简单版)

题意:

给一个长度为 n n n 的正整数数组 a a a,一个长度为 m m m 的查询数组 q q q, q [ i ] = ( v a l , m i n l e n , m a x l e n ) q[i] = (val, minlen, maxlen) q[i]=(val,minlen,maxlen)。 请你按输入顺序处理这 m m m 次查询,对于第 i i i 次查询 q [ i ] = ( v a l , m i n l e n , m a x l e n ) q[i] = (val, minlen, maxlen) q[i]=(val,minlen,maxlen): 请你输出是否存在一个 a a a 的子数组 s s s 满足 m i n ( s ) ≥ v a l min(s) \ge val min(s)≥val 且 s s s 的长度在 m i n l e n minlen minlen 和 m a x l e n maxlen maxlen 之间。

思路:

  • 首先,预处理除当前元素自身外,最小前缀元素的下标位置,和,最小后缀元素的下标位置,首尾第一个元素分别为-1和n;

  • 然后,对于每个元素x,我们要找出,当x作为某个连续子序列中元素的最小值时,该子序列最长为多少;

  • 通过预处理出的l和r元素,x, x, x, l, x, i, x, x, x, r, x, x,对于元素i,前缀最小为l,则l及之前的序列不能包含,否则会覆盖i作为子序列的最小值,同理,r及其之后的也不能涵盖,则当 a i a_i ai作为某个连续子序列中元素的最小值时, 最长为(r - l - 1)的子序列大小;

  • 依次处理出上述序列后,我们需要进行后缀最值处理,因为长度为n的序列最大的最小值成立,则长度为n-1上同样成立, 从而覆盖未处理过的区间大小;

    比如5 5 5 5 5,每个区间都能扩展5 5 5 5 5,覆盖的只有大小为5的区间长度,对于1 2 3 4的区间长度,通过取后缀最大即可覆盖;

  • 最后得到每个区间大小的最大的最小值;

AC code:

c 复制代码
void solve() {
    int n; cin >> n;
    vector<int> a(n);
    for (int i = 0; i < n; i ++) cin >> a[i];
    int mn = *min_element(a.begin(), a.end()), mx = *max_element(a.begin(), a.end());
    
    vector<int> mp(n + 1, -1);
    vector<int> l(n), r(n);
    deque<int> dq;
    for (int i = 0; i < n; i ++) { 
        while (!dq.empty() && a[dq.back()] >= a[i])
            dq.pop_back();
        l[i] = (dq.empty()) ? -1 : dq.back();
        dq.push_back(i);
    }
    
    dq.clear();
    for (int i = n - 1; i >= 0; i --) { 
        while (!dq.empty() && a[dq.back()] >= a[i])
            dq.pop_back();
        r[i] = (dq.empty()) ? n : dq.back();
        dq.push_back(i);
    }

    for (int i = 0; i < n; i ++) {
        int len = r[i] - l[i] - 1;
        mp[len] = max(mp[len], a[i]);
    }  
    for (int i = n - 1; i >= 0; i --) {
        mp[i] = max(mp[i], mp[i + 1]);
    }
    int m; cin >> m;
    while (m --) {
        int s, l, r; cin >> s >> l >> r;
        if (s <= mn) cout << "Yes" << endl;
        else if (s > mx) cout << "No" << endl;
        else {
            if (mp[l] >= s) cout << "Yes" << endl;
            else cout << "No" << endl;
        }
    }
}
相关推荐
野渡拾光1 小时前
【考研408数据结构-05】 串与KMP算法:模式匹配的艺术
数据结构·考研·算法
tainshuai3 小时前
用 KNN 算法解锁分类的奥秘:从电影类型到鸢尾花开
算法·分类·数据挖掘
Coovally AI模型快速验证9 小时前
农田扫描提速37%!基于检测置信度的无人机“智能抽查”路径规划,Coovally一键加速模型落地
深度学习·算法·yolo·计算机视觉·transformer·无人机
mit6.8249 小时前
[openvela] Hello World :从零开始的完整实践与问题复盘
c++·嵌入式硬件
pusue_the_sun9 小时前
数据结构:二叉树oj练习
c语言·数据结构·算法·二叉树
RaymondZhao3410 小时前
【全面推导】策略梯度算法:公式、偏差方差与进化
人工智能·深度学习·算法·机器学习·chatgpt
zhangfeng113310 小时前
DBSCAN算法详解和参数优化,基于密度的空间聚类算法,特别擅长处理不规则形状的聚类和噪声数据
算法·机器学习·聚类
啊阿狸不会拉杆11 小时前
《算法导论》第 32 章 - 字符串匹配
开发语言·c++·算法
小学生的信奥之路11 小时前
洛谷P3817题解:贪心算法解决糖果分配问题
c++·算法·贪心算法
曙曙学编程12 小时前
stm32——GPIO
c语言·c++·stm32·单片机·嵌入式硬件