题目链接:http://poj.org/problem?id=1015

题目:

题解:

我们考虑设计DP状态(因为这很显然是一个完全背包问题不是吗?)

dp[j][k]表示在外层循环到i时,选了j个人,此时辩方总分和控方总分差值为k的时,辩方和控方的总分的和的最大值

dp[j][k+a[i]-b[i]] = max (dp[j][k + a[i] - b[i]] , dp[j][k] + a[i] + b[i])

因为是完全背包,所以我们需要写一个search函数判断当前转移的状态是否已经选过了i,这样我们需要记录一个d[j][k]数组表示在dp[j][k]的状态是选哪一个人转移过来的,这恰好也是题目要求我们处理出来的

题解其实就是上述部分了

题外话:

我们考虑不写那个search函数,然后把j这一维倒序循环,根据《算法竞赛进阶指南》,这样的做法是可以的,但是笔者尝试之后还做不到,碰到的问题就是假设dp[j][k+a[i]-b[i]]被dp[j-1][k]转移得到,此时我们可以确定dp[j-1][k]这个状态没有选择i这个人。

但是在之后的循环中,dp[j-1][k]可能会被再次更新,我们要是只是统计答案的话这并没有什么影响,但是我们在d数组回溯的时候就会得到错误的转移,因为dp[j-1][k]可能被i更新,于是d[j-1][k]变成了i。补充:但是在我的AC代码里,显然是可以避免这种情况的,因为dp[j-1][k]不可能再被更新

除此之外,我发现i循环放在最外面是不行的。这是为什么?若是有读者知道请在评论区留言。

AC代码(加了search函数)

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std; int n,m,fix,time;
int a[],b[],dp[][],d[][],id[];
inline int read()
{
char ch=getchar();
int s=,f=;
while (ch<''||ch>'') {if (ch=='-') f=-;ch=getchar();}
while (ch>=''&&ch<='') {s=(s<<)+(s<<)+ch-'';ch=getchar();}
return s*f;
}
bool search(int j,int k,int i)
{
while (j&&d[j][k]!=i)
{
int o=d[j][k];
k-=a[o]-b[o];
j--;
}
if (j) return false;
return true;
}
int main()
{
while ()
{
n=read();m=read();
if (!n) break;
for (int i=;i<=n;i++)
{
a[i]=read();b[i]=read();
}
memset(dp,-,sizeof(dp));
memset(d,,sizeof(d));
fix=m*;
dp[][fix]=;
for (int j=;j<=m;j++)
for (int k=;k<=fix<<;k++)
if (dp[j-][k]>=)
{
for (int i=;i<=n;i++)
if (dp[j-][k]+a[i]+b[i]>dp[j][k+a[i]-b[i]]&&search(j-,k,i))
{
dp[j][k+a[i]-b[i]]=dp[j-][k]+a[i]+b[i];
//if (j==3&&k+a[i]-b[i]==57&&k==58) printf("sdfs%d\n",d[j-1][k]);
d[j][k+a[i]-b[i]]=i;
//if (j==2&&k+a[i]-b[i]==58) printf("%d\n",i);
}
}
int k,div;
for (int i=;i<=fix;i++)
if (dp[m][fix-i]!=-||dp[m][fix+i]!=-) {k=i;break;}
//printf("%d\n",k);
div=dp[m][fix-k]>dp[m][fix+k]?(fix-k):(fix+k);
printf("Jury #%d\n",++time);
printf("Best jury has value %d for prosecution and value %d for defence:\n",(dp[m][div]+div-fix)/,(dp[m][div]-div+fix)/);
int r=div;
for (int i=;i<=m;i++)
{
id[i]=d[m-i+][r];
r-=a[id[i]]-b[id[i]];
}
sort(id+,id++m);
for (int i=;i<=m;i++) printf(" %d",id[i]);
printf("\n\n");
}
return ;
}

最新文章

  1. AngularJS API
  2. 再谈 $* 和 $@ 在 Bash 中的表现
  3. Openstack Day1简介及虚拟环境搭建
  4. 浅析NRF51822合并文件之app_valid_setting_apply
  5. Spark RDD到底是个什么东西
  6. R(一): R基础知识
  7. Django编写RESTful API(三):基于类的视图
  8. PHPExcel-1.8导出
  9. Volatile的作用
  10. 【NOIP2013TG】solution
  11. 数值分析:Hermite多项式
  12. redis的持久化之RDB的配置和原理
  13. 记一次JAVA WEB项目解决XSS攻击的办法(亲测有效)
  14. 2019.03.28 bzoj3595: [Scoi2014]方伯伯的Oj(splay+map+set)
  15. SSM环境的搭建
  16. laravel get和all区别
  17. iOS开发设计多个target
  18. Hadoop集群配置(最全面总结 )(转)
  19. SecureCRT中常用linux命令 -《转载》
  20. 将 nginx 安装成 windows 的方法

热门文章

  1. 电子签章盖章之jQuery插件jquery.zsign
  2. 改动购物项图书数量的Ajax处理
  3. Android Java 程序员必备开发工具
  4. Yocto tips (19): Yocto SDK Toolchian的使用
  5. hdu 5335 Walk Out 搜索+贪心
  6. java中的system.out.println()和JSP中out.println()差别
  7. 在openwrt上编译一个最简单的ipk包
  8. SQL Server查询数据库空间分配情况、数据库备份信息
  9. Visual Studio蛋疼问题解决(1)
  10. (转载) TextView使用一些小技巧