方伯伯的玉米田

例题:方伯伯的玉米田

一位大佬的链接(有关二维树状数组)

前言

关于我做这道题的心路历程:

 1.我在某个晚上准备练习DP题,看到方伯伯的玉米田这
 道题,难度适宜,知识简单,准备做一做。一个晚上过去
 了,我除了推出了他每一次会抬高最右边的玉米外,还推
 出了一大堆错误结论,例如一定会从已有的上升序列中的
 左端点抬高啦。甚至还想,这不会是贪心题吧.
 2.第二个晚上,我搞出了一大堆DP方程,就剩最后怎么
 处理高度的问题。结果这个状态转移怎么也想不出。最后
 看题解,发现越看越糊涂,无聊之中点开了一个题解的链
 接。这是我转机的开始(要是没点开,可能我一辈子都做
 不出)
    首先我发现他的题解写错了 然后我发现我把题看错了。

接下来是关于做这道题的思路

不少大佬的题解都一开始说明f[i][j]指的是第i个点被覆盖j次的最大不下降序列。这其实就是这道题的枚举思路,然而并不需要写在代码中(Ps.很多题都是这样),这也就要是说,dp方程与他无关。

这就会有疑问:为什么无关?

还记得背包优化吗?

滚动数组使得一维消失了,因而从DP方程看上去,与枚举到哪一个物品无关。本题同样如此。这也使得在枚举剩余维度是一定要注重顺序。

接下来,将用到性质一:每次考虑抬高一段区间,我们都视这段区间的右端点为n

证明:

考虑从 l 开始抬高,当我们抬高到r,会对原有的上升子序列有两个影响
1. 1-l的序列中的每一个数,它右边比它大的数与抬
高前相比,有增多的趋势。我们称这对产生最长不下降子序列
有有利影响
2. r-n的每一个数,他前边比它小的数有减少趋势。我们
称这对产生最长不下降子序列有不利影响。

这就可以发现,我们把r定为n时,没有不利影响。

这就相当于,我们到一个点,就可以枚举它被抬高了几次。我们把区间的抬高转化为了点的抬高。

性质2:当前枚举到的点i,在它为j次抬高时,以他为终点的不下降子序列由前边任意一个比它抬高后矮的点x+1取max得来

这个看上去容易得到,其实把这个想通了,这道题做了一半。
首先,在实际写法中,需测判这个点x抬高次数是否小于j;
否则对于这样一组数据 4 1 2 1 4 3会出错。

实际上有了这两个性质后,我们就可以初步列出方程

f[i][j][k]=maxp=1i1f[p][j][k](j+k<j+k,k<k) f [ i ] [ j ] [ k ] = max p = 1 i − 1 f [ p ] [ j ′ ] [ k ′ ] ( j ′ + k ′ < j + k , k ′ < k )

可以看出这是一个三维转移方程,然而,我们发现
1. 我们大的高度总是由小的高度推出
1. 我们不关注每个位置具体的情况,我们只要求得最终答案即可。

介于上述两点,我们可以考虑降维。但我们从大的高度枚举到小的高度时,因为对每个位置的枚举是有序,先枚举的大高度又不会影响后枚举的小高度。
转移方程即为下述

f[j][k]=maxj+k<j+kf[j][k](k<k) f [ j ] [ k ] = max j ′ + k ′ < j + k f [ j ′ ] [ k ′ ] ( k ′ < k )

做到这里,已有了一个欠优的代码,考虑求max,可用二维树状数组优化后,降到 (log2n)2 ( l o g 2 n ) 2

关于二维树状数组,前面的大佬链接有详细的解释,也可以看我的另一篇博客。我们可以类比一维。一维树状数组可以让我们在 log2n l o g 2 n 的时间进行修改与查询线状的一条前缀和,二维树状数组能让我们在 (log2n)2 ( l o g 2 n ) 2 内进行一个矩形面积的查询。

#include
#define ll long long
#define RG register
#define IL inline
#define MAXN 10000
using namespace std;
IL int read(){
    RG int data=0;
    RG char ch=0;
    while(ch<'0'||ch>'9')ch=getchar();
    while(ch<='9'&&ch>='0'){
        data=data*10+ch-'0';
        ch=getchar();
    }
    return data;
}
IL int lowbit(int k){return k&(-k);}
int n,k,mh;
int btree[MAXN][MAXN],h[MAXN];
void fix(RG int ii,RG int jj,RG int x)
{
    for(RG int i=ii;i<=mh+k;i+=lowbit(i))
    for(RG int j=jj;j<=k+1;j+=lowbit(j))
    btree[i][j]=max(x,btree[i][j]);
}
int query(RG int ii,RG int jj){
    RG int ret=0;
    for(RG int i=ii;i;i-=lowbit(i))
    for(RG int j=jj;j;j-=lowbit(j))
    ret=max(ret,btree[i][j]);
    return ret;
}
int main()
{
    n=read(),k=read();
    //cout<
    for(RG int i=1;i<=n;++i){
        h[i]=read();
        mh=max(mh,h[i]);
    }
    RG int ans=0;
    for(RG int i=1;i<=n;++i){
        for(RG int j=k;j>=0;--j)
        {
            RG int x=query(h[i]+j,j+1)+1;
            ans=max(ans,x);
            fix(h[i]+j,j+1,x);
        }
    }
    printf("%d",ans);
}

你可能感兴趣的:(题解,dp,树状数组)