豆豆和豆沙正在分一些玩具,每个玩具有一个好玩值,每个人可以拿走任意数量的玩具,获得的愉快度为最小的好玩值。现在豆豆先拿,每个人轮流操作,直到没有玩具可以拿。豆豆想知道他能比豆沙多出多少愉快度?
题解:
首先题意可以转化为一个有序数组分成n段分别加减加减依次进行,然后考虑dp,因为要满足最优子结构,所以从前往后用
dp[i]
表示从i开始最大贡献,每次dp减去前面最小的(因为两个人都很聪明)。。
#include<bits/stdc++.h>
using namespace std;
struct IO{
streambuf *ib,*ob;
inline void init(){
ios::sync_with_stdio(false);
cin.tie(NULL);cout.tie(NULL);
ib=cin.rdbuf();ob=cout.rdbuf();
}
inline int read(){
char ch=ib->sbumpc();int i=0,f=1;
while(!isdigit(ch)){if(ch=='-')f=-1;ch=ib->sbumpc();}
while(isdigit(ch)){i=(i<<1)+(i<<3)+ch-'0';ch=ib->sbumpc();}
return i*f;
}
inline void W(int x){
static int buf[50];
if(!x){ob->sputc('0');ob->sputc('\n');return;}
if(x<0){ob->sputc('-');x=-x;}
while(x){buf[++buf[0]]=x%10;x/=10;}
while(buf[0]){ob->sputc(buf[buf[0]--]+'0');}
ob->sputc('\n');
}
}io;
const int Maxn=1e6+50;
int n,a[Maxn],dp[Maxn],mn;
int main(){
io.init();n=io.read();
for(int i=1;i<=n;i++)a[i]=io.read();
sort(a+1,a+n+1);
if(n<=1){io.W(a[1]);}
mn=a[1];
for(int i=1;i<=n;i++){
dp[i]=mn;mn=max(mn,a[i+1]-dp[i]);
}
io.W(dp[n]);
}