洛谷传送门

$Sol$

这是一道具有多个“体积维度”的$0/1$背包问题。

把$N$个候选人看做$N$个物品,那么每个物品有如下三种体积:

1.“人数”,每个候选人的“人数”都是$1$,最终要填满容积为$M$的背包

2.“辩方得分”,$a[i]$

3.“反方得分”,$b[i]$

要求的是辩方总分$D$和控方总分$P$的差的绝对值$|D-P|$最小,如果选择方法不唯一,那么在从中选择$D+P$最大的方案

所以将$a[i]-b[i]$作为体积之一,将$a[i]+b[i]$作为该物品的价值

注意这里的$a[i]-b[i]$可能为负,所以统一加上$4000$

以考虑到第$i$个人为阶段,$f[j][k]$表示已经在前$i$个人中选择了$j$个人,此时两方的总分之差为$k$时,两方的总分最大值。

$f[j][k]=max(f[j][k],f[j-1][k-(a[i]-b[i])]+a[i]+b[i])$

由于还需要输出方案,所以再加一个数组$d[i][j][k]$表示$f[i][j][k]$的第$i$个候选人选没选,这样就记录了方案。

$Code$

 #include<iostream>
#include<cstdio>
#include<cstring>
#define Rg register
#define il inline
#define mem(a,b) memset(a,b,sizeof(a));
#define go(i,a,b) for(Rg int i=a;i<=b;i++)
#define yes(i,a,b) for(Rg int i=a;i>=b;i--)
using namespace std;
il int read()
{
int x=,y=;char c=getchar();
while(c<''||c>''){if(c=='-')y=-;c=getchar();}
while(c>=''&&c<=''){x=(x<<)+(x<<)+c-'';c=getchar();}
return x*y;
}
int T,n,m,hhh=,t,ct,a[],b[],f[][],as[];
bool d[][][];
int main()
{
while()
{
n=read(),m=read();if(!n)break;
printf("Jury #%d\n",++T);
go(i,,n)a[i]=read(),b[i]=read();
mem(f,-);mem(d,);
f[][hhh]=;
go(i,,n)
yes(j,m,)
go(k,-,)
{
if(k-(a[i]-b[i])>=- && k-(a[i]-b[i])<= && f[j-][k-(a[i]-b[i])+hhh]+a[i]+b[i]>= && f[j][k+hhh]<f[j-][k-(a[i]-b[i])+hhh]+a[i]+b[i])
f[j][k+hhh]=f[j-][k-(a[i]-b[i])+hhh]+a[i]+b[i],d[i][j][k+hhh]=;
//if(f[j][k+hhh]>=0)cout<<"i:"<<i<<" j:"<<j<<" k:"<<k<<" f:"<<f[j][k+hhh]<<" d:"<<d[i][j][k+hhh]<<endl;
}
go(i,,)
{
if(f[m][hhh+i]>=){if(f[m][hhh+i]>f[m][hhh-i])t=i;else t=-i;break;}
else if(f[m][hhh-i]>=){t=-i;break;}
}
printf("Best jury has value %d for prosecution and value %d for defence:\n",(f[m][hhh+t]+t)/,(f[m][hhh+t]-t)/);
int nw=n;ct=m;
while(nw>)
{
int dd=d[nw][ct][t+hhh];
if(dd)as[ct]=nw,ct--,t-=(a[nw]-b[nw]);
nw--;
}
go(i,,m)printf(" %d",as[i]);printf("\n\n");
}
return ;
}

随机推荐

  1. jQuery Css类

    通过 jQuery,可以很容易地对 CSS 元素进行操作 jQuery 操作 CSS jQuery 拥有若干进行 CSS 操作的方法.我们将学习下面这些: addClass() - 向被选元素添加一个 ...

  2. 8-2udp和tcp网络编程以及粘包和解决粘包的方法

    一  tcp网络编程 server 端 import socket sk=socket.socket() #实例化一个对象 sk.setsockopt(socket.SOL_SOCKET,socket ...

  3. SVN的使用与教程

    1.先下载SVN安装包 SVN安装教程

  4. NetBeans配置

    NetBeans下载链接:https://netbeans.org/downloads/8.2/ 选择PHP×64版本 NetBeans 安装插件Darcula LAF for NetBeansctr ...

  5. Project Euler Problem 16-Power digit sum

    直接python搞过.没啥好办法.看了下别人做的,多数也是大数乘法搞过. 如果用大数做的话,c++写的话,fft优化大数乘法,然后快速幂一下就好了.

  6. 添加SuperSocket的启动代码到 Windows Azure 的 WorkRole 项目

    与其它SuperSocket程序相同,启动代码同样也要写到程序的入口处,如 Windows Azure 的 WorkRole 项目的OnStart() 方法: public override bool ...

  7. HTML5的5个的新特性

    特性1:正则表达式 无须使用服务端的检测,使用浏览器的本地功能,就可以帮助你判断邮箱的格式,URL,电话,防止用户输入错误的信息,通过H5的pattern属性,很方便的整合这个功能. <inpu ...

  8. C#的选择语句练习(二)

    1.输入a,b,c三个数,计算一元二次方程ax²+bx+c的根:若a=0,则不是一元二次方程:△=b²-4ac,根的计算公式为-b±√b²-4ac/2a:若△=b²-4ac>0,则方程有两个不一 ...

  9. Python--day62--ORM的使用

    4.Django里ORM的使用 1,手动创建数据库 2,在settings.py里面,配置数据库的连接信息 3,在项目/__init__.py告诉Django用pymysql模块代替MySQLdb(不 ...

  10. Python--day61--Django ORM单表操作之展示用户列表

    user_list.html views.py 项目的urls.py文件