罗大神说这题很简单,,,,然而我着实写的很难过。。。


题目链接:

http://acm.hust.edu.cn/vjudge/contest/view.action?cid=110495#problem/K

题意:

给定n个人对罪犯的d值和p值,从中选m个人使得abs(sum(d)−sum(p))最小,如果有多种情况则输出sum(p)+sum(d)最大的情况,并升序输出选择的人的编号。

分析:

这题是既要考虑差又要考虑和。还是绝对值最小。。刚看见有点懵逼。

其实不方,这题d和p最大只有20,m最大又只有20,就是说差的绝对值最大只有20 * 20,

那么我们设dp[i][j]为已经选出i个人,sum(p)−sum(d)为j的最大sum(p)+sum(d)。dp[i][j]=−1 表示这个状态无法达到。

那么问题来了,绝对值怎么处理,如果sum(p)−sum(d)小于0怎么办。。。

我们知道差的绝对值最大为20 * 20,那么我们给每个差都加上20 * 20 ,保证j大于等于0,这样就相当于以20 * 20 为原点,最终只要找到距离20 * 20 最近的dp值大于等于0的点就好了。。

想到这里就差不多了。。

很容易得到状态转移方程。

对于每个i和j的状态枚举给定的n个人,注意判断之前当前这个人之前是否出现过。

代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;
#define sa(a) scanf("%d", &a)
#define sal(a) scanf("%I64d", &a)
const int maxn = 1000 + 5, maxm = 200 + 5, INF = 0x3f3f3f3f;
int p[maxm], d[maxm];
struct DP{int a; int now;};
DP dp[20 + 5][maxn];
int tt[maxn];
int cnt;
void output(int he, int ans)
{
if(he == 0) return;
int index = dp[he][ans].now;
tt[he--] = index;
output(he, ans - p[index] + d[index]);
}
bool vis(int i, int j, int k)
{
while(i >= 0){
int index = dp[i][j].now;
if(index == k) return true;
i--;
j -= p[index] - d[index];
}
return false;
}
int main (void)
{
int n, m;
int cnt = 1;
while(scanf("%d%d", &n, &m) && n + m){
for(int i = 0; i < n; i++){
sa(p[i]); sa(d[i]);
}
int t = m * 20;
for(int i = 0; i <= m; i++){
for(int j = 0; j <= 2 * t; j++){
dp[i][j].a = -1;
dp[i][j].now = -1;
}
}
dp[0][t].a = 0;
for(int i = 0; i < m; i++){
for(int j = 0; j <= t * 2; j++){
if(dp[i][j].a < 0) continue;//状态无法到达
for(int k = 0; k < n; k++){
if(dp[i][j].a + p[k] + d[k] > dp[i + 1][j + p[k] - d[k]].a){
if(vis(i, j, k)) continue;//之前出现过
dp[i + 1][j + p[k] - d[k]].a = dp[i][j].a + p[k] + d[k];
dp[i + 1][j + p[k] - d[k]].now = k;
}
}
}
}
int i = 0;
while(dp[m][t + i].a < 0 && dp[m][t - i].a < 0) i++;
int ans;
if(dp[m][t + i].a > dp[m][t - i].a) ans = t + i;
else ans = t - i;
printf("Jury #%d\n", cnt++);
printf("Best jury has value %d for prosecution and value %d for defence:\n",
(ans - t + dp[m][ans].a)/2, (dp[m][ans].a - ans + t)/2);
output(m, ans);
sort(tt + 1, tt + m + 1);
for(int i = 1; i <= m; i++)
printf(" %d%c", tt[i] + 1, i == m?'\n':' ');
printf("\n");
}
return 0;
}

POJ 1015 Jury Compromise【DP】的更多相关文章

  1. POJ 1015 Jury Compromise(dp坑)

    提议:在遥远的国家佛罗布尼亚,嫌犯是否有罪,须由陪审团决定.陪审团是由法官从公众中挑选的.先随机挑选n个人作为陪审团的候选人,然后再从这n个人中选m人组成陪审团.选m人的办法是:控方和辩方会根据对候选 ...

  2. 背包系列练习及总结(hud 2602 && hdu 2844 Coins && hdu 2159 && poj 1170 Shopping Offers && hdu 3092 Least common multiple && poj 1015 Jury Compromise)

    作为一个oier,以及大学acm党背包是必不可少的一部分.好久没做背包类动规了.久违地练习下-.- dd__engi的背包九讲:http://love-oriented.com/pack/ 鸣谢htt ...

  3. POJ 1015 Jury Compromise(双塔dp)

    Jury Compromise Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 33737   Accepted: 9109 ...

  4. OpenJudge 2979 陪审团的人选 / Poj 1015 Jury Compromise

    1.链接地址: http://bailian.openjudge.cn/practice/2979 http://poj.org/problem?id=1015 2.题目: 总Time Limit: ...

  5. POJ 3280 Cheapest Palindrome【DP】

    题意:对一个字符串进行插入删除等操作使其变成一个回文串,但是对于每个字符的操作消耗是不同的.求最小消耗. 思路: 我们定义dp [ i ] [ j ] 为区间 i 到 j 变成回文的最小代价.那么对于 ...

  6. poj1015 Jury Compromise【背包】

    Jury Compromise Time Limit: 1000MS   Memory Limit: 65536K Total Submissions:32355   Accepted:8722   ...

  7. poj 1015 Jury Compromise(背包+方案输出)

    \(Jury Compromise\) \(solution:\) 这道题很有意思,它的状态设得很...奇怪.但是它的数据范围实在是太暴露了.虽然当时还是想了好久好久,出题人设了几个限制(首先要两个的 ...

  8. POJ 1015 Jury Compromise dp分组

    第一次做dp分组的问题,百度的~~ http://poj.org/problem?id=1015 题目大意:在遥远的国家佛罗布尼亚,嫌犯是否有罪,须由陪审团决定.陪审团是由法官从公众中挑选的.先随机挑 ...

  9. poj 1015 Jury Compromise(背包变形dp)

    In Frobnia, a far-away country, the verdicts in court trials are determined by a jury consisting of ...

随机推荐

  1. JS实现让其他浏览器自动转至微信浏览器打开链接

    //判断是否是pc function is_pc() { var sUserAgent = navigator.userAgent.toLowerCase(); var bIsIpad = sUser ...

  2. SEO 第四章

    SEO第四章 课程目标: 掌握网站TDK的优化方法 1.  页面TKD介绍 Title  keywords description 标题 关键字 描述 网站的每一个页面都有三大标签(主页.栏目页.内容 ...

  3. (译文)IOS block编程指南 4 声明和创建blocks

    Declaring and Creating Blocks (声明和创建blocks) Declaring a Block Reference (声明一个block引用) Block variable ...

  4. tomcat 安全配置文档

    1.配置文档中使用$CATALINA_HOME变量声明为tomcat的安装目录并明确写出了tomcat的配置文件路径,此路径为测试环境的路径,线上系统对应配置文件的路径可能不一样,在进行相关配置时,应 ...

  5. Python基础3 函数 变量 递归 -DAY3

    本节内容 1. 函数基本语法及特性 2. 参数与局部变量 3. 返回值 嵌套函数 4.递归 5.匿名函数 6.函数式编程介绍 7.高阶函数 8.内置函数 温故知新 1. 集合 主要作用: 去重 关系测 ...

  6. Convert Sorted List to Balanced Binary Search Tree leetcode

    题目:将非递减有序的链表转化为平衡二叉查找树! 参考的博客:http://blog.csdn.net/worldwindjp/article/details/39722643 利用递归思想:首先找到链 ...

  7. 核心动画中的几种layer

    第10章其他有用的层 免责申明(必读!):本博客提供的所有教程的翻译原稿均来自于互联网,仅供学习交流之用,切勿进行商业传播.同时,转载时不要移除本申明.如产生任何纠纷,均与本博客所有人.发表该翻译稿之 ...

  8. [IOS初学]ios 第一篇 storyboard 与viewcontroller的关系 - Zoe_J

    时间 2014-07-27 16:08:00  博客园-所有随笔区 原文  http://www.cnblogs.com/zoe-j/p/3871501.html 主题 StoryBoard 学习了一 ...

  9. 洛谷——P4932 浏览器

    P4932 浏览器 __stdcall给了你n个点,第i个点有权值x[i],对于两个点u和v,如果x[u] xor x[v]的结果在二进制表示下有奇数个1,那么在u和v之间连接一个Edge,现在__s ...

  10. cf上分的失落->高兴->更失落

    cf昨日做出一个题居然div2打了1800多名直接上分了...我原以为垂直落地但是....我现在1399差一分就能蓝名了啊啊啊啊,以后不一定会有这个水平的啊,给个蓝名体验卡不行吗,多加一分会死啊... ...