题意: 在井中有n个矮子,每个矮子都有ai(脚到肩膀的高度)和bi(手臂长度), 当存在A1 + A2 + ... + Ak-1 + Ak + Bk >= H,k可以从井里面逃出去。问最多能逃出去几个人。
解题思路:按ai+bi和的值从小到大排序后dp,dp[i][j]表示前i个矮子跑出去j个时,需要之前井中剩下的人的最小A高度之和。有如下转移方程dp[i][j] = min(dp[i-1][j] - s[i-1].a, max(dp[i-1][j-1], H - sumA[i-1] - s[i-1].b ))。最后找到最大的j满足dp[n][j]<=0即为所求。
粗略证明:当逃出去的组合(逃出去的人编号的组合)是一个的时候,如果某个组合的人可以都逃出去,先跑ai+bi小的肯定可以跑出去。
#include<iostream>
#include<stdio.h>
#include<algorithm>
const int N=3000;
const int inf=1e9;
using namespace std;
struct node
{
int a,b;
void read(){ scanf("%d%d",&a,&b); }
bool operator <(const node tmp)const { return a+b<tmp.a+tmp.b; }
}no[N];
int opt[N][N],n,h;
int solve()
{
int sum=0;
for(int i=1;i<=n;i++) sum+=no[i].a;
for(int j=1;j<=n;j++) opt[0][j]=inf;
opt[0][0]=0;
for(int i=1;i<=n;i++)
{
opt[i][0]=0;
for(int j=1;j<=n;j++){
opt[i][j]=opt[i-1][j];
if(sum-opt[i-1][j-1]+no[i].b>=h)
opt[i][j]=min(opt[i][j],opt[i-1][j-1]+no[i].a);
// cout<<i<<' '<<j<<' '<<opt[i][j]<<' '<<opt[i-1][j]<<endl;
}
}
for(int j=n;j>=0;j--)
if(opt[n][j]!=inf)
return j;
}
int main()
{
while(scanf("%d",&n)!=-1)
{
for(int i=1;i<=n;i++)
no[i].read();
scanf("%d",&h);
sort(no+1,no+n+1);
cout<<solve()<<endl;
}
return 0;
}