CF505C

Mr. Kitayuta, the Treasure Hunter - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

一眼为DP

该如何考虑dp状态?显然到了 第 i 个点的时候,还需要知道达到此时走的步的大小,才能进行dp转移

考虑dp[i][j]为这次走了j大步走到i能获得最多的宝藏,但这回MLE

考虑优化空间大小,显然位置是不容易优化的,尽量优化第二维数组的空间(这是本题的难点也是核心)

最坏的结果是第一步 为 1,则 1+2+3+4+...+n>30001 → n=260,最大可能步数也仅仅为260

对于其他第一步大于1的,最后最大的步数只可能更小,最小步数逆这来同理。

所以对于任何步数 x,d为题目给出的的第一步,则x-d∈[-300,300]

令M=300,所以x-d+M属于[0,2*M],这种数据范围就很适合作为第二维dp的空间大小

 

枚举时,第一重循环 i 依旧为第一维度,第二重循坏 j 为第二维度,j=x-d+M;

则 x=j+d-M;根据题目要求要保证x>=1;

转移方程:

dp[i+x][j]=max(dp[i+x][j],dp[i][j]+a[i+x]) (i+x<=m&&x>=1)

dp[i+x+1][j+1]=max(dp[i+x+1][j+1],dp[i][j]+a[i+x+1]) (i+x+1<=m&&x>=1)

dp[i+x-1][j-1]=max(dp[i+x-1][j-1],dp[i][j]+a[i+x-1])  (i+x-1<=m&&x>1)

初始化:x=d,dp[d][M]=a[d],其余都为-INF;

Code:

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define mp make_pair
#define pb push_back   
#define popb pop_back  
#define fi first
#define se second
#define popcount __builtin_popcount
#define popcountll __builtin_popcountll
const int N=30010;
const int M=300;
const int inf=1e9;
//const ll INF=1e18;
int T,n,m,d,a[N];
int dp[N][800];
bool vis[N][800];
vector<int> v[N];
inline int read()
{
    int x=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=(x<<1)+(x<<3)+(ch^48);ch=getchar();}
    return x*f;
}
int main()
{
//    freopen("","r",stdin);
//    freopen("","w",stdout);
//    ios::sync_with_stdio(0);
//    cin.tie(0);
    n=read(),d=read();
    for(int i=1;i<=n;i++)
    {
        int x=read();
        m=max(m,x);
        a[x]++;
    }
    for(int i=0;i<=m;i++)
        for(int j=0;j<=700;j++)
            dp[i][j]=-inf; 
    dp[d][M]=a[d];
    for(int i=d;i<=m;i++)
        for(int j=0;j<=2*M;j++)
        {
            int x=j+d-M;
            if(x<1) continue;
            if(x-1&&i+x-1<=m) dp[i+x-1][j-1]=max(dp[i+x-1][j-1],dp[i][j]+a[i+x-1]);
            if(i+x<=m) dp[i+x][j]=max(dp[i+x][j],dp[i][j]+a[i+x]);
            if(i+x+1<=m) dp[i+x+1][j+1]=max(dp[i+x+1][j+1],dp[i][j]+a[i+x+1]);
        }
    int ans=0;
    for(int i=d;i<=m;i++)
        for(int j=0;j<=2*M;j++)
            ans=max(ans,dp[i][j]);
    printf("%d",ans);
    return 0;
}

 

posted @ 2023-05-08 19:28  QAQ啥也不会  阅读(12)  评论(0编辑  收藏  举报