HDU-1171-Big Event in HDU

题目链接

http://acm.hdu.edu.cn/showproblem.php?pid=1171

题目大意: n种设备,每种设备价值为v 数量为m  希望 以最小的差距分成两堆   第一堆不能比第二堆少,

 

开始看完这题让我想起了小时候与朋友一起去摘野果后分野果,就是我先拿一个,拿最大的,朋友再拿一个,也剩下的拿最大的,以此推,不过这样的话测试数据都过不了,

让我知道以前看是公平的分野果的方法其实并不是最好的方法。

 

这题我用的是多重背包的的思路:多重背包可行性问题么,以价值为费用,以总价值的一半为背包容量。

代码

#include<stdio.h>
#include<string.h>
#include<iostream>
#include<algorithm>
using namespace std;

int dp[125005];

int main(void)
{
int n,i,j,k;
int v[55],m[55];
while(scanf("%d",&n)==1&&n>=0)
{
memset(dp,0,sizeof(dp));
int s=0;
for(i=0;i<n;i++)
{
scanf("%d%d",&v[i],&m[i]);
s=s+v[i]*m[i];
}
int w=s/2;
for(i=0;i<n;i++)
{
for(j=0;j<=w;j++)
{
for(k=0;k<=m[i]&&k*v[i]<=j;k++)
{
dp[j]=max(dp[j],dp[j-k*v[i]]+k*v[i]);
}
}
}
printf("%d %d\n",s-dp[w],dp[w]);
}
return 0;
}

 

参数

1218MS  780K  733 B    

 

这题可以把多重背包改为01背包

 

#include<stdio.h>
#include<string.h>
#include<iostream>
using namespace std;

int v[5005];
int dp[125005];

int main(void)
{
int i,j,k,n,s;
int a,b;
while(scanf("%d",&n)==1&&n>=0)
{
k=0; s=0;
memset(dp,0,sizeof(dp));
for(i=0;i<n;i++)
{
scanf("%d%d",&a,&b);
for(j=0;j<b;j++)
{
v[k++]=a;
s=s+a;
}
}
int w=s/2;
for(i=0;i<k;i++)
{
for(j=w;j>=v[i];j--)
{
dp[j]=max(dp[j],dp[j-v[i]]+v[i]);
}
}
printf("%d %d\n",s-dp[w],dp[w]);
}
return 0;
}

 

 

参数

765MS 796K 709 B

快一些,省时。

 

刚开始接触背包问题  思维有点难以形成,慢慢来。

 

posted @ 2014-11-16 15:21  立刻行动  阅读(133)  评论(0编辑  收藏  举报