题目链接:

http://poj.org/problem?id=1112

Team Them Up!

Time Limit: 1000MS
Memory Limit: 10000K
#### 问题描述
> Your task is to divide a number of persons into two teams, in such a way, that:
>
> everyone belongs to one of the teams;
>
> every team has at least one member;
>
> every person in the team knows every other person in his team;
>
> teams are as close in their sizes as possible.
>
> This task may have many solutions. You are to find and output any solution, or to report that the solution does not exist.
#### 输入
> For simplicity, all persons are assigned a unique integer identifier from 1 to N.
>
> The first line in the input file contains a single integer number N (2 If the solution to the problem does not exist, then write a single message "No solution" (without quotes) to the output file. Otherwise write a solution on two lines. On the first line of the output file write the number of persons in the first team, followed by the identifiers of persons in the first team, placing one space before each identifier. On the second line describe the second team in the same way. You may write teams and identifiers of persons in a team in any order.
####样例输入
> 5
> 2 3 5 0
> 1 4 5 3 0
> 1 2 5 0
> 1 2 3 0
> 4 3 2 1 0

样例输出

3 1 3 5

2 2 4

题意

给你若干个人,要把他们分成两组,其中同一组内要保证任意两个人互相认识(题目给的边都是单向边),求一个使得两组人数最接近的方案,要求输出一个可行分组方案。

题解

首先,把关系图建处理(对于(u,v)如果不存在(v,u)相当于这条边不存在),处理出补图,然后题目就转换成了一个二分图问题了,跑下黑白染色,判断是否有可行解,如果有,则用dp跑下,求出最优解就ok啦。dp[i][j]表示已经处理了i个联通分量(跑黑白染色会处理出若干个联通分量),其中一组能凑出j个人的方案。

代码

#include<map>
#include<set>
#include<cmath>
#include<queue>
#include<stack>
#include<ctime>
#include<vector>
#include<cstdio>
#include<string>
#include<bitset>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<functional>
#include<sstream>
using namespace std;
#define X first
#define Y second
#define mkp make_pair
#define lson (o<<1)
#define rson ((o<<1)|1)
#define mid (l+(r-l)/2)
#define sz() size()
#define pb(v) push_back(v)
#define all(o) (o).begin(),(o).end()
#define clr(a,v) memset(a,v,sizeof(a))
#define bug(a) cout<<#a<<" = "<<a<<endl
#define rep(i,a,b) for(int i=a;i<(b);i++)
#define scf scanf
#define prf printf typedef long long LL;
typedef vector<int> VI;
typedef pair<int,int> PII;
typedef vector<pair<int,int> > VPII; const int INF=0x3f3f3f3f;
const LL INFL=0x3f3f3f3f3f3f3f3fLL;
const double eps=1e-8;
const double PI = acos(-1.0); //start---------------------------------------------------------------------- const int maxn=111; int n;
bool gra[maxn][maxn];
bool dp[maxn][maxn];
VI lis[maxn][3];
int color[maxn],tot;
//黑白染色
bool dfs(int u) {
for(int v=1; v<=n; v++) {
if(!gra[u][v]) continue;
if(!color[v]) {
color[v]=3-color[u];
lis[tot][color[v]].pb(v);
bool su=dfs(v);
if(!su) return false;
} else {
if(color[v]==color[u]) return false;
}
}
return true;
} VI ans[3];
bool used[maxn];
//处理出最优方案
void print(int i,int j) {
if(i==0) {
return;
}
int a=lis[i][1].sz(),b=lis[i][2].sz();
if(j>=a&&dp[i-1][j-a]) {
print(i-1,j-a);
rep(ii,0,lis[i][1].sz()) {
ans[1].pb(lis[i][1][ii]);
}
} else if(j>=b&&dp[i-1][j-b]) {
print(i-1,j-b);
rep(ii,0,lis[i][2].sz()) {
ans[1].pb(lis[i][2][ii]);
}
}
} int main() {
while(scf("%d",&n)==1&&n) {
clr(gra,0);
for(int i=1; i<=n; i++) {
int v;
while(scf("%d",&v)==1&&v) {
gra[i][v]=1;
}
}
for(int i=1; i<=n; i++) {
for(int j=1; j<i; j++) {
gra[i][j]=gra[j][i]=!(gra[i][j]&&gra[j][i]);
}
} //黑白染色
clr(color,0);
rep(i,0,maxn) lis[i][1].clear(),lis[i][2].clear();
tot=0;
bool su=1;
for(int i=1; i<=n; i++) {
if(!color[i]) {
++tot;
color[i]=1;
lis[tot][1].pb(i);
su=dfs(i);
}
if(!su) break;
}
if(!su) {
prf("No solution\n");
continue;
} //01背包,不是选和不选,而是选1还是选2.
clr(dp,0);
dp[0][0]=1;
for(int i=1; i<=tot; i++) {
int a=lis[i][1].sz(),b=lis[i][2].sz();
for(int j=0; j<=100; j++) {
if(j>=a&&j>=b) {
dp[i][j]=dp[i-1][j-a]|dp[i-1][j-b];
} else if(j>=a) {
dp[i][j]=dp[i-1][j-a];
} else if(j>=b) {
dp[i][j]=dp[i-1][j-b];
}
}
} //输出最优方案
int Mi=INF,pos=-1;
for(int i=1; i<=100; i++) {
if(dp[tot][i]) {
if(Mi>abs(i-(n-i))) {
Mi=abs(i-(n-i));
pos=i;
}
}
} ans[1].clear(),ans[2].clear();
clr(used,0);
print(tot,pos); rep(i,0,ans[1].sz()) used[ans[1][i]]=1;
for(int i=1; i<=n; i++) {
if(!used[i]) ans[2].pb(i);
} sort(all(ans[1]));
sort(all(ans[2])); for(int i=1; i<=2; i++) {
prf("%d ",ans[i].sz());
rep(j,0,ans[i].sz()) prf("%d%c",ans[i][j],j==ans[i].sz()-1?'\n':' ');
}
}
return 0;
} //end----------------------------------------------------------------------- /*
3
2 0
1 3 0
2 0
*/

POJ 1112 Team Them Up! 二分图判定+01背包的更多相关文章

  1. HDU 5313 Bipartite Graph(二分图染色+01背包水过)

    Problem Description Soda has a bipartite graph with n vertices and m undirected edges. Now he wants ...

  2. DP:Cow Exhibition(POJ 2184)(二维问题转01背包)

        牛的展览会 题目大意:Bessie要选一些牛参加展览,这些牛有两个属性,funness和smartness,现在要你求出怎么选,可以使所有牛的smartness和funness的最大,并且这两 ...

  3. POJ 2923 Relocation (状态压缩,01背包)

    题意:有n个(n<=10)物品,两辆车,装载量为c1和c2,每次两辆车可以运一些物品,一起走.但每辆车物品的总重量不能超过该车的容量.问最少要几次运完. 思路:由于n较小,可以用状态压缩来求解. ...

  4. poj 2184 Cow Exhibition(dp之01背包变形)

    Description "Fat and docile, big and dumb, they look so stupid, they aren't much fun..." - ...

  5. 图论+dp poj 1112 Team Them Up!

    题目链接: http://poj.org/problem?id=1112 题目大意: 有编号为1~n的n个人,给出每个人认识的人的编号,注意A认识B,B不一定认识A,让你将所有的人分成两组,要求每组的 ...

  6. POJ 2923 Relocation(状压DP+01背包)题解

    题意:给你汽车容积c1,c2,再给你n个包裹的体积,问你最少运几次能全运走 思路:用2进制表示每次运送时某物在不在此次运送之中,1在0不在.我们把运送次数抽象成物品价值,把状态抽象成体积,用一个dp[ ...

  7. POJ 2184 Cow Exhibition 奶牛展(01背包,变形)

    题意:有只奶牛要证明奶牛不笨,所以要带一些奶牛伙伴去证明自己.牛有智商和幽默感,两者可为负的(难在这),要求所有牛的智商和之 / 幽默感之和都不为负.求两者之和的最大值. 思路:每只牛可以带或不带上, ...

  8. POJ 1948 Triangular Pastures【二维01背包】

    题意:给出n条边,用这n条边构成一个三角形,求三角形的最大面积. 先求面积,用海伦公式,s=sqrt(p*(p-a)*(p-b)*(p-c)),其中a,b,c分别为三角形的三条边,p为三角形的半周长, ...

  9. POJ 3624 Charm Bracelet(01背包裸题)

    Charm Bracelet Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 38909   Accepted: 16862 ...

随机推荐

  1. GATK--数据预处理,质控,检测变异

    版权声明:本文源自 解螺旋的矿工, 由 XP 整理发表,共 13781 字. 转载请注明:从零开始完整学习全基因组测序(WGS)数据分析:第4节 构建WGS主流程 | Public Library o ...

  2. FSCapture注册码

    企业版序列号: name:bluman serial/序列号/注册码:VPISCJULXUFGDDXYAUYF

  3. sqlplus 中 各列对齐设定

    设置列标题的对齐方式 JUSTFIFY  {L[EFT]|C[ENTER]|R[IGHT]} SQL> col ename justify center SQL> /

  4. CF248E Piglet's Birthday

    题面 题意翻译 给定$n$个货架,初始时每个上面有$a[i]$个蜜罐. 有$q$次操作,每次操作形如$u,v,k$,表示从货架$u$上任意选择$k$个蜜罐试吃(吃过的也还能吃),吃完后把这$k$个蜜罐 ...

  5. c#总结:datatable的方法大全

    最近在做一个数据处理,保存到datatable中,在过程中了解了datatable一些用法,总结一下: //定义 DataTable dt = new DataTable(); //写入列名: htT ...

  6. Codeforces 469 D. Two Sets (并查集)

    题目链接:Two Sets 题意: 有n个数,要分成A.B两组,要求如果x∈A则a-x∈A,如果x∈B则b-x∈B,问是否存在一种符合要求的分法. 题解: 并查集,先增加两个点表示A和B集合的根,对于 ...

  7. [Luogu1891]疯狂LCM[辗转相减法]

    题意 多组询问,每次给定 \(n\) ,求:\(\sum_{i=1}^nlcm(i,n)\) . \(\rm T \leq 3\times 10^4\ ,n \leq 10^6\). 分析 推式子: ...

  8. CSS快速入门-定位(position)

    一.概述 CSS 定位 (Positioning) 属性允许你对元素进行定位. 定位的基本思想很简单,它允许你定义元素框相对于其正常位置应该出现的位置,或者相对于父元素.另一个元素甚至浏览器窗口本身的 ...

  9. JUC——延迟队列

    所谓的延迟队列最大的特征是它可以自动通过队列进行脱离,例如:现在有一些对象被临时保存着,但是有可能该集合对象是一个公共对象,那么里面的某些数据如果不在使用的时候就希望其可以在指定的时间达到后自动的消失 ...

  10. csb反编译为csd,并自动进行资源的删除

    好多人都想将csb进行反编译为csd,然后进行资源的清理 目前自己的项目也遇到了类似的问题,所以进行了整理 还有很多不完善的地方,后续会一步步加深 请大家多多指教 下载链接:https://pan.b ...