陪审团(01背包, 难)

在一个遥远的国家,一名嫌疑犯是否有罪需要由陪审团来决定。

陪审团是由法官从公民中挑选的。

法官先随机挑选N个人(编号1,2…,N)作为陪审团的候选人,然后再从这N个人中按照下列方法选出M人组成陪审团。

首先,参与诉讼的控方和辩方会给所有候选人打分,分值在0到20之间。

第 i 个人的得分分别记为p[i]和d[i]。

为了公平起见,法官选出的M个人必须满足:辩方总分D和控方总分P的差的绝对值|D-P|最小。

如果选择方法不唯一,那么再从中选择辨控双方总分之和D+P最大的方案。

求最终的陪审团获得的辩方总分D、控方总分P,以及陪审团人选的编号。

注意:若陪审团的人选方案不唯一,则任意输出一组合法方案即可。
输入格式

输入包含多组测试数据。

每组测试数据第一行包含两个整数N和M。

接下来N行,每行包含两个整数p[i]和d[i]。

每组测试数据之间隔一个空行。

当输入数据N=0,M=0时,表示结束输入,该数据无需处理。
输出格式

对于每组数据,第一行输出’Jury #C’,C为数据编号,从1开始。

第二行输出“Best jury has value P for prosecution and value D for defence:”,P为控方总分,D为辩方总分。

第三行输出按升序排列的陪审人选编号,每个编号前输出一个空格。

每组数据输出完后,输出一个空行。
数据范围

1≤N≤200
,
1≤M≤20
0≤p[i],d[i]≤20

输入样例:

4 2
1 2
2 3
4 1
6 2
0 0

输出样例:

Jury #1
Best jury has value 6 for prosecution and value 4 for defence:
2 3

思路:本题较难, 但可以看出是01背包的模子, 即前i个选j个, 但体积和价值发生了变化,价值可以用评价分数之和和之差来评定,对于每一个状态来说,可以用f(i, j, k, m) 来恒定。i为前i个人,j为选了j个人了, k为差值, m 为总和。可以想到用k来作为背包的一个维度(相当于体积),细节见于代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int n, m, d[205], p[205];
int f[21][801]; //向右平移400
bool choose[205][21][801];
void print(int i, int j, int k) {
    if (j==0) return;
    if (choose[i][j][k+400]) {
        print(i-1,j-1,k-(d[i]-p[i]));
        printf(" %d", i);
    } else {
        print(i-1,j,k);
    }
}
int main() {
    int T = 0;
    while (cin>>n>>m && n) {
        for(int i=1;i<=n;i++)
            scanf("%d%d",&p[i],&d[i]);
        memset(f, 0xcf, sizeof(f));
        f[0][0+400]=0;
        for(int i=1;i<=n;i++)
            for(int j=m;j;j--)
                for(int k=-20*m;k<=20*m;k++) {
                    choose[i][j][k+400] = false; // 默认不选
                    if (k-(d[i]-p[i])<-20*m || k-(d[i]-p[i])>20*m) continue;
                    if (f[j][k+400] < f[j-1][k-(d[i]-p[i])+400]+d[i]+p[i]) { // 选
                        f[j][k+400] = f[j-1][k-(d[i]-p[i])+400]+d[i]+p[i];
                        choose[i][j][k+400] = true;
                    }
                }
        int delta=1<<30;
        int raw_delta;
        int sum = 0;
        int ansk;
        for(int k=-20*m;k<=20*m;k++)
            if (f[m][k+400] >= 0 && (abs(k)<delta || abs(k)==delta && f[m][k+400]>sum)) // f[m][k]合法 
                delta=abs(k), raw_delta=k, sum=f[m][k+400], ansk=k;
        // d-p=raw_delta
        // d+p=sum
        printf("Jury #%d\n", ++T);
        printf("Best jury has value %d for prosecution and value %d for defence:\n", (sum-raw_delta)/2, (sum+raw_delta)/2);
        print(n, m, ansk);
        puts("\n");
    }
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值