题目链接:

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. 求助:将以下ES5格式代码转换为ES6格式!!!

    function Slider(id){     //属性     //  1. 通过id获取元素对象(大盒子)     this.bigBox = document.getElementById(i ...

  2. js遍历添加栏目类添加css 再点击其它删除css

    //js遍历添加栏目类添加css 再点击其它删除css $(".radio-group .ckselect").each(function(index) { $(this).cli ...

  3. 新版u-boot移植到s3c2440开发板(一)--建立单板

    由于没有系统的学习shell,所以Makefile大多数看不懂,一个小小的细节,把我难住了几天.现在开始分享我的操作过程 本文所有linux下的操作是root用户,如果你使用普通用户,请在命令前加上 ...

  4. lua虚拟机概述

    何为虚拟机 用于模拟计算机运行的程序.是个中间层,它处于脚本语言和硬件之间的一个程序.每一门脚本语言都会有自己定义的opcode("操作码"),可以理解为这门程序自己定义的&quo ...

  5. springboot快速入门(五)——事务管理

    一.入门 概念就不再赘述了,由于一般我们是通过service控制事务,这里给出注解式的示例: package com.example.demo; import com.example.demo.bea ...

  6. Spring第四天——SSH整合

    (从整合开始,使用回归使用eclipse) 一.三大框架版本:  struts2 hibernate5 spring4 二.SSH三大框架回顾: Hibernate: ORM思想 核心配置文件: 单独 ...

  7. 20145234黄斐《java程序设计》第十三周代码检查

    在IDEA中对P145 MathTool.java 使用JUnit进行单元测试,测试用例不少于三个,要包含正常情况,边界情况. 提交测试代码和运行结果截图,加上学号水印,提交码云代码链接. 码云链接  ...

  8. 对Oracle的游标进行更为精细的控制

    摘要自 Oracle性能控制艺术 DECLARE l_ename emp.ename%TYPE :='SCOTT'; l_empno emp.empno%TYPE; l_cursor INTEGER; ...

  9. 面试题-python基础

    一.Python基础 1.什么是python?使用python有什么好处? python是一种编程语言,它有对象.模块.线程.异常处理和自动内存管理.它简洁,简单.方便.容易扩展.有许多自带的数据结果 ...

  10. .net core中automapper的使用

    automapper 是将两个类中的相同字段进行映射,也可以指定字段进行映射:将 UserDao的id 映射为 User 的age CreateMap<UserDao, User>() . ...