HDU 6406 Taotao Picks Apples 杭电多校第八场
本文最后更新于 2097 天前,其中的信息可能已经有所发展或是发生改变。

内容纲要

题意

题解

由于这是一个不用维护原有序列的操作,我们可以考虑维护一个ST表进行RMQ查询。
这个RMQ是用来干什么呢?
根据题目数据范围,我们知道这道题不能容忍$\mathcal O(n^2)$及以上的算法
我们必须要在$\mathcal O(nlogn)$的时间复杂度下完成所有的操作。
维护一个dp数组,$dp[cur] = dp[first_greater_than_cur] + 1$
这个时候就需要我们用RMQ了。二分$[cur,n]$,找到第一个比当前元素大的元素的位置,这时dp值就可以加1.
贪心预处理原序列的选择情况、每个位置选择的长度以及prev数组用于维护上一个被选择的元素。
接下来是查询部分
我们需要对查询进行分类讨论。
查询有以下几种情况:
1.修改的元素是被选择的序列中的元素。
这时候我们要考虑这个数相对于上一个数的大小的情况
那么这时候出现一个特殊情况,就是这个数是第一个数。
这个时候$ans = 1 + dp[first_element_greater_than_cur]$

若不是第一个数,则是分两种情况考虑。
– 1. 比当前的数上一个数小或相等
– 2. 比上一个数大

对于第一种情况,$ans = cnt[prev] + dp[first_element_greater_than_prev]$
而第二种情况则是 $ans = cnt[p] + dp[first_element_greater_than_cur]$

2.修改的元素不在被选择的序列中
也分两种情况考虑。
– 1. 比当前位置的上一个被选元素小
这时候修改的数对答案没有贡献,因此不需要考虑
– 2. 比当前位置的上一个被选元素大
这时候则是$ans = cnt[prev] + 1 + dp[first_element_greater_than_cur]$,因为这个元素必选 + 上一个元素以及之前被选数的个数 + 下一个第一个比当前元素大的元素对应的dp值就是答案


AC代码

#pragma GCC optimize("O3")
#include <iostream>
#include <algorithm>
#include <vector>
#include <array>
#include <cstring>
#include <queue>
#include <regex>

using namespace std;
using ll = long long;
using lll = __int128;
using pii = pair<int,int>;

const int maxn = 2e5+6;
int n,m,tot;
int h[maxn];
int st[maxn][32-__builtin_clz(maxn)];
int dp[maxn];
bool select[maxn];
int pre[maxn],cnt[maxn];

inline void init_st()
{
    int l = 31 - __builtin_clz(n);
    for(auto i = 0;i<n;++i)st[i][0] = h[i];
    for(int j = 0;j<l;++j) {
        for(int i = 0;i<(1 + n - (1 << j));++i) {
            st[i][j+1] = max(st[i][j],st[i + (1 << j)][j]);
        }
    }
}

int rmq(int l,int r)
{
    int k = 31 - __builtin_clz(r - l + 1);
    return max(st[l][k], st[r - (1 << k) + 1][k]);
}

int first_greater_than_after(int pos,int val)
{
    int l = pos + 1,r = n,mid;
    while(l < r) {
        mid = (l + r ) >> 1;
        if (rmq(pos + 1, mid) <= val) l = mid + 1;
        else r = mid;
    }
    return l;
}

void prepare() 
{
    int _cnt = 1;
    dp[n] = 0;
    for(int i = n - 1;i >= 0;--i) {
        int pos = first_greater_than_after(i,h[i]);
        dp[i] = dp[pos] + 1;
    }
    select[0] = true;
    pre[0] = -1;
    cnt[0] = 1;
    int last = 0,prevh = h[0];
    for(int i = 1;i<n;++i) {
        pre[i] = last;
        if (h[i] > prevh) {
            select[i] = true;
            last = i;
            prevh = h[i];
            ++_cnt;
        }
        else {
            select[i] = false;
        }
        cnt[i] = _cnt;
    }
    /*
    for(int i = 0;i<n;++i)
    {
        cout << "select["<<i<<"]="<<select[i]<<endl;
    }
    */
    tot = _cnt;
}


int main()
{
    ios::sync_with_stdio(false);
    cin.tie(0);
    cout.tie(0);
    int T;
    cin >> T;
    while(T--)
    {
        cin >> n >> m;
        for(int i = 0;i<n;++i) cin >> h[i];
        init_st();
        prepare();
        /*
        for(int i = 0;i<n;++i)cout << "dp["<<i<<"]="<<dp[i]<<endl;
        for(int i = 0;i<n;++i)cout << "cnt["<<i<<"]="<<cnt[i]<<endl;
        */
        for(int i = 0;i<m;++i)
        {
            int p,q;
            cin >> p >> q;
            --p;
            if(select[p])
            {
                if(p == 0)
                {
                    // change first element
                    int ans = 1;
                    ans += dp[first_greater_than_after(p,q)];
                    cout << ans << '\n';
                }
                else
                {
                    // not first element
                    if(h[pre[p]] >= q)
                    {
                        //change value to smaller and 
                        //prev high is greater than or equal q
                        int ans = cnt[pre[p]];//ans from start to prev
                        int pos = first_greater_than_after(p,h[pre[p]]);
                        // first element greater than pos
                        cout << ans + dp[pos] << '\n';
                    }
                    else
                    {
                        //change value not smaller or equal than prev
                        //check if there exists at least one element is greater than current modify value but smaller than prev value
                        int ans = cnt[p];
                        int pos = first_greater_than_after(p,q);
                        cout << ans + dp[pos] << '\n';
                    }
                }
            }
            else
            {
                // the element has not been selected
                if(h[pre[p]] >= q)
                {
                    // if prev element is greater or equal than current modify value
                    // it doesn't change the answer.
                    cout << tot << '\n';
                }
                else
                {
                    //cout << "came here" << endl;

                    // q is greater than prev element,may have effected to next element(s)
                    // search first element which greater than q
                    int ans = cnt[pre[p]] + 1;//select this value,so plus one
                    //cout << ans << endl;
                    int pos = first_greater_than_after(p,q);
                    //cout << pos << endl;
                    cout << ans + dp[pos] << '\n';
                }
            }
        }
    }
}
暂无评论

发送评论 编辑评论


				
|´・ω・)ノ
ヾ(≧∇≦*)ゝ
(☆ω☆)
(╯‵□′)╯︵┴─┴
 ̄﹃ ̄
(/ω\)
∠( ᐛ 」∠)_
(๑•̀ㅁ•́ฅ)
→_→
୧(๑•̀⌄•́๑)૭
٩(ˊᗜˋ*)و
(ノ°ο°)ノ
(´இ皿இ`)
⌇●﹏●⌇
(ฅ´ω`ฅ)
(╯°A°)╯︵○○○
φ( ̄∇ ̄o)
ヾ(´・ ・`。)ノ"
( ง ᵒ̌皿ᵒ̌)ง⁼³₌₃
(ó﹏ò。)
Σ(っ °Д °;)っ
( ,,´・ω・)ノ"(´っω・`。)
╮(╯▽╰)╭
o(*////▽////*)q
>﹏<
( ๑´•ω•) "(ㆆᴗㆆ)
😂
😀
😅
😊
🙂
🙃
😌
😍
😘
😜
😝
😏
😒
🙄
😳
😡
😔
😫
😱
😭
💩
👻
🙌
🖕
👍
👫
👬
👭
🌚
🌝
🙈
💊
😶
🙏
🍦
🍉
😣
Source: github.com/k4yt3x/flowerhd
颜文字
Emoji
小恐龙
花!
上一篇
下一篇