POJ 2438 Children’s Dining (哈密顿图模板题之巧妙建反图 )
Description
Usually children in kindergarten like to quarrel with each other. This situation annoys the child-care women. For instant, when diner time comes, a fierce conflict may break out when a certain couple of children sitting side by
side who are hostile with each other. Although there aren't too many children dining at the same round table, but the relationship of "enemy" or "friend" may be very complex. The child-care women do come across a big problem. Now it is time for you to help
them to figure out a proper arrangement of sitting, with which no two "enemy" children is adjacent.
Now we assume that there are 2 * n children who sit around a big table, and that none has more than n - 1 "enemies".
Input
The input is consisted of several test blocks. For each block, the first line contains two integers n and m (1 <= n <= 200, 0 <= m <= n (n - 1)). We use positive integers from 1 to 2 * n to label the children dining round table.
Then m lines followed. Each contains positive integers i and j ( i is not equal to j, 1 <= i, j <= 2 * n), which indicate that child i and child j consider each other as "enemy". In a input block, a same relationship isn't given more than once, which means
that if "i j" has been given, "j i" will not be given.
There will be a blank line between input blocks. And m = n = 0 indicates the end of input and this case shouldn't be processed.
Output
For each test block, if the proper arrangement exist, you should print a line with a proper one; otherwise, print a line with "No solution!".
Sample Input
1 0
2 2
1 2
3 4
3 6
1 2
1 3
2 4
3 5
4 6
5 6
4 12
1 2
1 3
1 4
2 5
2 6
3 7
3 8
4 8
4 7
5 6
5 7
6 8
0 0
Sample Output
1 2
4 2 3 1
1 6 3 2 5 4
1 6 7 2 3 4 5 8
分析:
本题给出的是小朋友之间的敌对关系,有敌对关系的不能挨着。所以建图的时候要初始化g[i][j]为1,如果相邻的话,就改为0,这样就建好图了。因为一共有2n个小朋友,但是每个人最多有n-1个敌人,所以每个小朋友可选择的人数肯定大于n+1,这就说明在建完图后,每个点的度数大于n+1,因此任意两个点的度数之和大于2n,满足哈密顿图的充分条件,所以一定存在哈密顿图。最后直接套上哈密顿图模板就能解出来了。
需要注意的一点就是m的值可能是为0 的,所有的小朋友彼此之家都没有敌对关系。
算法讲解
我们了解欧拉图谈论的实际上是图上关于边的可行遍性的问题,而哈密吨图的要求与点有关,图G的一个回路,若它通过图的每一个节点有且仅有一次,就是哈密顿回路。存在哈密顿回路的图就是哈密顿图。、
欧拉回路就是从图上的一点出发,经过所有的边必须且只能一次,最终回到原点的路径。
哈密顿回路就是从一点出发,经过所有的点必须且只能一次,最终回到起点的路径,图中有的边可以不经过,但是不会有边被经过两次。
满足哈密顿图的两个判定条件:
一:设图G是具有n个顶点的无向连通图,如果G中任意两个不同顶点的度数之和大于等于n,则G具有哈密顿回路,即G是哈密顿图。这是判断哈密顿图的充分条件,即不满足定理条件时也有可能存在哈密顿回路,图G也可能时哈密顿图,但是满足条件一定是哈密顿图。
二:如果图G<V,E>时哈密顿图,则对于V的任意一个非空子集,若以|S|表示S中元素的数目,G-S表示删除了S中的点以及这些所关联的边后得到的子图,则W(G-S)<=|S|成立。其中W(G-S)是G-S中联通分支数。这个判定方法上哈密顿图的必要条件,即如果不满足条件一丁不是哈密顿图,但满足条件还不能说这个图是哈密顿图。
构造哈密顿图的算法过程:
1.任意找两个相邻的节点S和T,在其基础上扩展出一条尽量长的没有重复节点的路径。也就是说,如果节点S与节点v相邻,而且节点v不在路径S-->T上,则可以把该路径变成v-->S-->T,然后把v变成新的S。从S和T分别向两头扩展,直到无法扩展为止,即所有与S或T相邻的节点都在S-->T上(这里的S和T是不断变化的)。
2.若S和T相邻,则路径S-->T形成一个回路。
3.若S和T不相邻,可以构造出一个回路。设路径S-->T上有k+2个节点,依次为S,v1,v2,···,vk和T。可以证明存在节点vi(i属于[1,k)),满足vi与T相邻,且v(i+1)与S相邻。找到上述节点vi,把原路径变成S-->vi-->T-->v(i+1),即形成一个回路。
4.现在有了一条没有重复节点的路径,如果其长度为n,则哈密顿回路就找到了。如果回路的长度小于n,由于整个图是连通的,所以在还回路上,一定存在一点与回路以外的点相邻。那么就把该回路从这一点处断开,就变成一条路径,同时还可以将与之相连的点加入路径。在按照步骤1的方法尽量扩展路径,则一定有新的节点被加进来。接着回到步骤2。
代码:
#include<stdio.h>
#include<iostream>
#include<string.h>
using namespace std;
int n,m;
int tu[409][409];
int vis[409];
int ans[409];
void init()
{
for(int i=0; i<=n; i++)
for(int j=0; j<=n; j++)
{
if(i==j)
tu[i][j]=0;
else
tu[i][j]=1;
}
memset(vis,0,sizeof(vis));
memset(ans,0,sizeof(ans));
}
void reverse(int ans[409],int s,int t)//反转函数,s和t分别表示反转开始和结束的界限
{
int temp;
while(s<t)
{
temp=ans[s];
ans[s]=ans[t];
ans[t]=temp;
s++;
t--;
}
}
void Hamilton()//哈密顿图模板
{
int s=1,t;//初始化取s为1号点
int ansi=2;
int i,j,w,temp;
for(i=1; i<=n; i++)
{
if(tu[s][i]==1)
break;//只要任意找到一个和起始点相邻的点就行
}
t=i;//这个点作为当前的结束点
vis[s]=vis[t]=1;
ans[0]=s;
ans[1]=t;//存储当前的起点和终点
while(true)
{
while(true)//从t点开始向外扩展
{
for(i=1; i<=n; i++)
{
if(tu[t][i]==1&&vis[i]==0)//找到任意的一个就可以向外扩
{
ans[ansi++]=i;
vis[i]=1;
t=i;//将t的值改变后接着往下找
break;
}
}
if(i>n) break;//相当于从t没法向外扩展了
}
//将新的到的序列倒置,s和t互换,这样接着从t开始向外扩展,相当于将原来的s向外扩展
w=ansi-1;
i=0;
reverse(ans,i,w);
//将s和t互换
temp=s;
s=t;
t=temp;
//从新的t开始向外扩展,相当于原来的s向外扩展,代码与上面的原理一样
while(true)
{
for(i=1; i<=n; i++)
{
if(tu[t][i]==1&&vis[i]==0)
{
ans[ansi++]=i;
vis[i]=1;
t=i;
break;
}
}
if(i>n)break;
}
//如果s和t不相邻的话,进行调整
if(tu[s][t]==0)
{
//取序列中的一点i,值得ans[i]与t相邻,ans[i+1]与s相邻
for(i=1; i<ansi-2; i++)
{
if(tu[ans[i]][t]==1&&tu[s][ans[i+1]]==1)
break;
}
//将从ans[i+1]到t的部分倒置
w=ansi-1;
i++;
t=ans[i];
reverse(ans,i,w);
}
//如果s和t相邻
if(ansi==n) return ;//当前序列中包含n个元素,算法结束
//当前序列元素个数小于n,寻找ans[i],使得ans[i]与ans外的一点相邻
for(j=1; j<=n; j++)
{
if(vis[j]) continue;
for(i=1; i<ansi-2; i++)
{
if(tu[ans[i]][j])
break;
}
if(tu[ans[i]][j])
break;
}
s=ans[i-1];
t=j;
//将ans中的s到ans[i-1]部分的ans倒置
reverse(ans,0,i-1);
//将ans中的ans[i+1]到t部分的ans倒置
reverse(ans,i,ansi-1);
//将j加入到尾部
ans[ansi++]=j;
vis[j]=1;
}
}
int main()
{
int u,v;
while(~scanf("%d%d",&n,&m),n|m)
{
n=n*2;
init();
for(int i=0; i<m; i++)
{
scanf("%d%d",&u,&v);
tu[u][v]=tu[v][u]=0;//建立反图,表示这两个小朋友不能坐在一起,用图理解的话就是这两个点不相邻
}
Hamilton();
printf("%d",ans[0]);
for(int i=1; i<n; i++)
printf(" %d",ans[i]);
printf("\n");
}
return 0;
}
POJ 2438 Children’s Dining (哈密顿图模板题之巧妙建反图 )的更多相关文章
- poj 2438 Children's Dining
http://poj.org/problem?id=2438 题意: 有2*N个人要坐在一张圆桌上吃饭,有的人之间存在敌对关系,安排一个座位次序,使得敌对的人不相邻. 假设每个人最多有N-1个敌人.如 ...
- POJ 2438 Children's Dining(哈密顿回路)
题目链接:http://poj.org/problem?id=2438 本文链接:http://www.cnblogs.com/Ash-ly/p/5452615.html 题意: 有2*N个小朋友要坐 ...
- POJ 1273 - Drainage Ditches - [最大流模板题] - [EK算法模板][Dinic算法模板 - 邻接表型]
题目链接:http://poj.org/problem?id=1273 Time Limit: 1000MS Memory Limit: 10000K Description Every time i ...
- POJ 1860 Currency Exchange【bellman-Ford模板题】
传送门:http://poj.org/problem?id=1860 题意:给出每两种货币之间交换的手续费和汇率,求出从当前货币s开始交换回到s,能否使本金增多. 思路:bellman-Ford模板题 ...
- POJ 1258 Agri-Net 【Prime】模板题
题目链接>>> 题目大意: 给你N*N矩阵,表示N个村庄之间的距离.FJ要把N个村庄全都连接起来,求连接的最短距离(即求最小生成树).解析如下: #include <c ...
- POJ 3624 Charm Bracelet(01背包模板题)
题目链接 Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 52318 Accepted: 21912 Descriptio ...
- POJ 1273:Drainage Ditches 网络流模板题
Drainage Ditches Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 63339 Accepted: 2443 ...
- POJ 2186:Popular Cows Tarjan模板题
Popular Cows Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 25945 Accepted: 10612 De ...
- POJ 1258:Agri-Net Prim最小生成树模板题
Agri-Net Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 45050 Accepted: 18479 Descri ...
随机推荐
- node入门学习(一)
一.安装node.js 方式很多npm,git等,新手建议从官网上直接去下载node的安装包.一键安装. 二.创建一个web服务器. const http = require('http'); htt ...
- Powershell笔记之help about_XXX
<<about_Arrays>> 1. 一直好奇gettype()输出中的Name这一列的Object[]是什么意思,现在终于明白了,请看下面的例子: 2. GM的不同用法 $ ...
- 如何在Windows下查看JAVA端口占用情况(阿里面试)
如需要确定谁占用了9050端口 为例: 1.Windows平台 在windows命令行窗口下执行: 1.查看所有的端口占用情况 C:\>netstat -ano 协议 本地地址 外部地址 状态 ...
- 深入理解JAVA虚拟机阅读笔记6——线程安全与锁优化
线程安全:如果一个对象可以安全的被多个线程同时使用,那它就是线程安全的. 一.Java中的线程安全 1.不可变 不可变的对象一定是线程安全的.String.枚举类型.java.lang.Number的 ...
- C# 房贷计算器
设计背景 百度小程序中的房贷计算器不能满足我个人的需求,故而开发一个.NET小程序.希望后期能用JS重写,发布在网上供大家使用. 设计思路 根据百度公式:等额本息月还款 = [贷款本金×月利率×(1+ ...
- [提升性选讲] 树形DP进阶:一类非线性的树形DP问题(例题 BZOJ4403 BZOJ3167)
转载请注明原文地址:http://www.cnblogs.com/LadyLex/p/7337179.html 树形DP是一种在树上进行的DP相对比较难的DP题型.由于状态的定义多种多样,因此解法也五 ...
- Mybatis笔记六:Mybatis中SqlSessionFactoryBuilder/SqlSessionFactory/SqlSession/映射器实例的作用域(Scope)和生命周期
SqlSessionFactoryBuilder 这个类可以被实例化.使用和丢弃,一旦创建了 SqlSessionFactory,就不再需要它了.因此 SqlSessionFactoryBuilder ...
- 使用IDEA远程部署tomcat和调试
环境: CentOS 7 Tomcat 9.0.1 jdk-9.0.1 IntelliJ IDEA 2017.3 Tomcat中的配置 在catalina.sh文件中加入以下的配置 CATALINA_ ...
- 【BZOJ1502】【NOI2005】月下柠檬树
Portal 传送门 Solution 显然的是,每一个圆的影子,就是从树上的圆按光线方向平移至地面的圆.至于两个圆之间的连接部分,则是每两个在树上相邻的圆的,对应的影子圆的,公切线围起来的部分,如下 ...
- Android Studio 换主题 + 背景图片 + 去掉白色竖线
1.去掉AS编辑区域右边的白色竖线: 把right margin 设置的大一点就可以了,默认是120 ,设置成 1200就ok了 2.AS主题下载换装 可以去如下网站下载,然后导入jar, 具体用法百 ...