2023NOIP A层联测9-天竺葵

天竺葵/无法阻挡的子序列/很有味道的题目

我们称一个长度为 k k k 的序列 c c c 是好的,当且仅当对任意正整数 i i i [ 1 , k − 1 ] [1,k-1] [1,k1] 中,满足 c i + 1 > b i × c i c_{i+1}>b_i \times c_i ci+1>bi×ci b b b 序列在下文描述。

小 L 现在给你两个序列 a , b a,b a,b,你需要从 a a a 序列中找出一个最长的子序列 c c c,使得 c c c 是好的。

输出这个最长的子序列的长度即可。


暂且把这个问题叫做带权最长上升子序列。

显然,类似于求 L I S LIS LIS,如果我们在 a a a 序列的前 i i i 个数中已经选了一个好的序列 c c c,那么 c c c 的最后一个一定是最小的(因为后面更容易满足条件增加长度)。

于是用二分, l o w i low_i lowi 表示长度为 i i i 的带权最长上升子序列的 a i ⋅ b i a_i\cdot b_i aibi 的最小值。

每次用 lower_bound \texttt{lower\_bound} lower_bound l o w low low 中查找大于等于 a i a_i ai 的第一个位置,用 a i ⋅ b i a_i\cdot b_i aibi 更新该位置,同时记录答案。

这样就做完了。

细节详见代码。

#include
using namespace std;
typedef long long ll;
int n,ans=1;
ll a[1000001],b[1000001],low[1000001];
ll read()
{
    ll sum=0;
    int c=getchar();
    while(c<48||c>57) c=getchar();
    while(c>=48&&c<=57) sum=sum*10+c-48,c=getchar();
    return sum;
}
int main()
{
    freopen("C.in","r",stdin);
    freopen("C.out","w",stdout);
    n=read();
    for(int i=1;i<=n;i++) a[i]=read();
    for(int i=1;i<=n;i++) b[i]=read();
    memset(low,0x3f,sizeof(low));
    for(int i=1;i<=n;i++){
        int czn=lower_bound(low+1,low+1+ans,a[i])-low;
        ans=max(ans,czn);
        low[czn]=min(a[i]*b[czn],low[czn]);
    }
    cout<<ans;
}

你可能感兴趣的:(算法,c++,数据结构)