2018.08.17 洛谷[POI2010]GRA-The Minima Game(线性dp)

传送门
短代码神奇dp。
自己yy的思路居然1A了好高兴啊!
不难想到每个人选择的时候一定是取连续的最大的那一段数,自然需要先排序。
然后可以用dp[i]表示当前最大数是a[i]的时候先手可以获得的最优值。

不难想到dp[i]跟dp[1]~dp[i-1]都有关系,其实就是dp[i]=max(a[j]dp[j1])dp[i]=max(a[j]−dp[j−1]),然后可以发现这个东西当j<ij<i时都在推dp[i-1]的时候求过一遍了,于是dp[i]=max(dp[i1],a[i]dp[i1])dp[i]=max(dp[i−1],a[i]−dp[i−1])

于是又多了一道短代码题。

代码:

#include<bits/stdc++.h>
#define ll long long
#define N 1000005
using namespace std;
inline ll read(){
    ll ans=0;
    char ch=getchar();
    while(!isdigit(ch))ch=getchar();
    while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
    return ans;
}
ll a[N],dp[N];
int n;
int main(){
    n=read();
    for(int i=1;i<=n;++i)a[i]=read();
    sort(a+1,a+n+1);
    dp[0]=0,dp[1]=a[1];
    for(int i=2;i<=n;++i){
        dp[i]=dp[i-1];
        dp[i]=max(dp[i],a[i]-dp[i-1]);
    }
    cout<<dp[n];
    return 0;
}

转载于:https://www.cnblogs.com/ldxcaicai/p/9738374.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值