POJ 1015 Jury Compromise dp分组
第一次做dp分组的问题,百度的~~
http://poj.org/problem?id=1015
题目大意:在遥远的国家佛罗布尼亚,嫌犯是否有罪,须由陪审团决定。陪审团是由法官从公众中挑选的。先随机挑选n个人作为陪审团的候选人,然后再从这n个人中选m人组成陪审团。选m人的办法是:控方和辩方会根据对候选人的喜欢程度,给所有候选人打分,分值从0到20。为了公平起见,法官选出陪审团的原则是:选出的m个人,必须满足辩方总分和控方总分的差的绝对值最小。如果有多种选择方案的辩方总分和控方总分的之差的绝对值相同,那么选辩控双方总分之和最大的方案即可。
2 2
1 2
3 4
大概思路如下。
设dp[i][j]表示选i个人,当前的差值是j的时候,最大的和值是dp[i][j]
由于他是abs的,这样会产生相同的j。但是应该分开的,因为路径是不同的,产生的dp[i][j]那个值不应该被其他覆盖。
然后就是记录路径,
path[i][j]表示是由那个值进化而来的,即可说是上一个,也可以说是这个的值。
然后思路就是,
枚举i : 1 to m //选m个人
k : 0 to 2 * fix //枚举差值。
判断即可。
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#define IOS ios::sync_with_stdio(false)
using namespace std;
#define inf (0x3f3f3f3f)
typedef long long int LL; #include <iostream>
#include <sstream>
#include <vector>
#include <set>
#include <map>
#include <queue>
#include <string>
const int fix = ;
int dp[][]; // dp[i][j]表示选了i个人,差值为j的时候,的最大和值
int path[][];
int D[ + ];
int P[ + ];
int sum[ + ];
int dis[ + ];
set<int>ans;
int n, m;
void init() {
memset(dp, -, sizeof dp);
memset(path, -, sizeof path);
ans.clear();
}
void work() {
init();
for (int i = ; i <= n; ++i) {
cin >> D[i] >> P[i];
dis[i] = D[i] - P[i];
sum[i] = D[i] + P[i];
}
dp[][fix] = ; //因为fix了,所以0的时候,其实就是fix的时候
for (int i = ; i <= m; ++i) {
for (int k = ; k <= * fix; ++k) {
if (dp[i - ][k] >= ) { //如果这个状态可行,才能再选举一个人进来
for (int j = ; j <= n; ++j) {
if (dp[i - ][k] + sum[j] > dp[i][k + dis[j]]) { //加了这个人后,最优解变大,然后差值就是变成k + dis[j]的
//还需要检查j是否在dp[i - 1][k]这个路径上了。
int t = path[i - ][k];
int pre = i - ;
int ff = k;
bool flag = true;
while (t != -) {
if (t == j) {
flag = false;
break;
}
ff -= dis[t]; //减去当前这个点的值
t = path[--pre][ff];
}
if (flag == true) {
dp[i][k + dis[j]] = dp[i - ][k] + sum[j];
path[i][k + dis[j]] = j;
}
}
}
}
}
}
int i = fix;
int j = ;
while (dp[m][i + j] < && dp[m][i - j] < ) { //排除没用的状态
++j;
}
int KK; //最小的差值 对应的那个dp,可以对应输出最大的和值
if (dp[m][i + j] > dp[m][i - j]) {
KK = i + j;
} else KK = i - j;
int t = path[m][KK];
int pre = m;
int ff = KK;
while (t != -) {
ans.insert(t);
ff -= dis[t];
t = path[--pre][ff];
}
int df = , pf = ;
for (set<int> :: iterator it = ans.begin(); it != ans.end(); ++it) {
df += D[*it];
pf += P[*it];
}
static int gg = ;
printf("Jury #%d\n", ++gg);
printf("Best jury has value %d for prosecution and value %d for defence:\n", df, pf);
for (set<int> :: iterator it = ans.begin(); it != ans.end(); ++it) {
cout << " " << *it;
}
cout << endl << endl;
} int main() {
#ifdef local
freopen("data.txt","r",stdin);
#endif
IOS;
while (cin >> n >> m && (n + m)) {
work();
}
return ;
}
POJ 1015 Jury Compromise dp分组的更多相关文章
- POJ 1015 Jury Compromise dp
大致题意: 从n个候选人中选出m个人作为陪审团.为了让陪审团的选择更公平,辩方和控方都为这n个候选人给出了满意度(辩方为D[j],控方为P[j],范围0至20).现在要使得选出的m位候选人的辩方总和与 ...
- 背包系列练习及总结(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 ...
- POJ 1015 Jury Compromise(双塔dp)
Jury Compromise Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 33737 Accepted: 9109 ...
- OpenJudge 2979 陪审团的人选 / Poj 1015 Jury Compromise
1.链接地址: http://bailian.openjudge.cn/practice/2979 http://poj.org/problem?id=1015 2.题目: 总Time Limit: ...
- poj 1015 Jury Compromise(背包+方案输出)
\(Jury Compromise\) \(solution:\) 这道题很有意思,它的状态设得很...奇怪.但是它的数据范围实在是太暴露了.虽然当时还是想了好久好久,出题人设了几个限制(首先要两个的 ...
- poj 1015 Jury Compromise(背包变形dp)
In Frobnia, a far-away country, the verdicts in court trials are determined by a jury consisting of ...
- POJ 1015 Jury Compromise(dp坑)
提议:在遥远的国家佛罗布尼亚,嫌犯是否有罪,须由陪审团决定.陪审团是由法官从公众中挑选的.先随机挑选n个人作为陪审团的候选人,然后再从这n个人中选m人组成陪审团.选m人的办法是:控方和辩方会根据对候选 ...
- POJ 1015 Jury Compromise【DP】
罗大神说这题很简单,,,,然而我着实写的很难过... 题目链接: http://acm.hust.edu.cn/vjudge/contest/view.action?cid=110495#proble ...
- HDU POJ 1015 Jury Compromise(陪审团的人选,DP)
题意: 在遥远的国家佛罗布尼亚,嫌犯是否有罪,须由陪审团决定.陪审团是由法官从公众中挑选的.先随机挑选n个人作为陪审团的候选人,然后再从这n个人中选m人组成陪审团.选m人的办法是:控方和辩方会根据对候 ...
随机推荐
- 集训Day1
雅礼集训2017Day1的题 感觉上不可做实际上还挺简单的吧 T1 区间加 区间除法向下取整 查询区间和 区间最小值 大力上线段树,把除法标记推到底,加法标记就是按照线段树的来 先拿30 然后60的数 ...
- 递归/非递归----python深度遍历二叉树(前序遍历,中序遍历,后序遍历)
递归代码:递归实现很简单 '二叉树结点类' class TreeNode: def __init__(self, x): self.val = x self.left = None self.righ ...
- poj3281Dining——网络流匹配
题目:http://poj.org/problem?id=3281 网络流做图中的匹配,注意为了限制每头牛只匹配一种食物和一种饮料,要把牛拆成两个点,之间连边权为1的边. 代码如下: #include ...
- Selenium实战脚本集—新浪微博发送QQ每日焦点
Selenium实战脚本集-新浪微博发送QQ每日焦点 http://www.spasvo.com/ceshi/open/kygncsgj/Selenium/201549150822.html 背景 很 ...
- HDU 4625. JZPTREE
题目简述:给定$n \leq 50000$个节点的数,每条边的长度为$1$,对每个节点$u$,求 $$ E_u = \sum_{v=1}^n (d(u, v))^k, $$ 其中$d(u, v)$是节 ...
- Robot FrameWork基础学习(四) 元素定位
元素定位 对于web自动化测试来说,就是操作页面的各种元素,在操作元素之间需要先找到元素,换句话说就是定位元素. Selenium2Library提供了非常丰富的定位器: 虽然提供了这么多种定位方式, ...
- 如何在模板类中使用这些point类型?
博客转载自:http://www.pclcn.org/study/shownews.php?lang=cn&id=271 由于PCL模块较多,并且是一个模板库,在一个源文件里包含很多PCL算法 ...
- 条款39:明智而审慎的使用private继承
Use private inheritance judiciously. 如果classes之间的继承关系是private,编译器不会自动将一个derived class对象转换为一个base cla ...
- FZU 2059 MM (并查集+排序插入)
Problem 2059 MM Accept: 109 Submit: 484Time Limit: 1000 mSec Memory Limit : 32768 KB Problem ...
- 初识Kotlin之集合
Kotlin的集合是让我为之心动的地方,丰富的高阶函数帮助我们高效开发.今天介绍Kotlin的基础集合用法.获取集合元素的函数.过滤元素的函数.元素排序的函数.元素统计的函数.集合元素映射的函数.集合 ...