对于第一组数据:同学1允许同学2或同学3在他之前拿到菜;同学2允许同学3在他之前拿到菜;同学3比较小气,他必须比他后面的同学先拿菜…… 一种最优的方案是按同学3、同学2、同学1、同学4、同学5做菜,每道菜所需的时间分别是0、8、1、6及1。 【数据规模和约定】对于30%的数据,满足1 ≤ N ≤ 20。对于100%的数据,满足1 ≤ N ≤ 1,000,0 ≤ Ti ≤ 1,000,0 ≤ Bi ≤ 7,1 ≤ C ≤ 5。存在30%的数据,满足0 ≤ Bi ≤ 1。存在65%的数据,满足0 ≤ Bi ≤ 5。存在45%的数据,满足0 ≤ Ti ≤ 130。
#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
int n,ans;
int f[1010][260][17],cnt[260],T[1010],B[1010];
inline int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<‘0‘||gc>‘9‘) {if(gc==‘-‘)f=-f; gc=getchar();}
while(gc>=‘0‘&&gc<=‘9‘) ret=ret*10+gc-‘0‘,gc=getchar();
return ret*f;
}
int calc(int a,int b)
{
if(a<=0) return 0;
return T[a]^T[b];
}
void work()
{
memset(f,0x3f,sizeof(f));
n=rd();
int i,j,k,l;
for(i=1;i<=n;i++) T[i]=rd(),B[i]=min(rd(),n-i);
f[0][0][8]=0;
for(i=0;i<n;i++) for(j=0;j<(1<<min(n-i,8));j++) for(k=max(8-i,0);k<=min(n-i,8)+8;k++)
{
if(j&1)
{
f[i+1][j>>1][k-1]=min(f[i+1][j>>1][k-1],f[i][j][k]);
continue;
}
int lim=min(n-i,8);
for(l=1;l<=lim;l++) if(!(j&(1<<(l-1))))
{
f[i][j|(1<<(l-1))][l+8]=min(f[i][j|(1<<(l-1))][l+8],f[i][j][k]+calc(i+k-8,i+l));
lim=min(lim,l+B[i+l]);
}
}
ans=1<<30;
for(i=max(8-n,0);i<=8;i++) ans=min(ans,f[n][0][i]);
printf("%d\n",ans);
}
int main()
{
int T=rd();
while(T--) work();
return 0;
}//1 2 5 0 4 0