定理1:在一个具有n个顶点的无向连通图G中,如果任意两个顶点的度数之和大于n,则G具有Hamilton回路。此条件为充分条件

定理2:设图G = <V,E>,是Hamilton图,则对于v的任意一个非空子集S,若以|S|表示S中元素数目,G-S表示G中删除了S中的点以及与这些点关联的边后得到的子图,则满足G-S的连通分支数W(G-S)<=|S|。此条件为必要条件。

构造Hamilton回路的算法过程,分成以下几个步骤:

1. 任意找两个相邻的节点 S 和 T,在它们基础上扩展出一条尽量长的没有重复节点的路径。也就是说,如果 S 与节点 v 相邻,而且 v 不在路径 S → T 上,则可以把该路径变成 v → S → T,然后 v 成为新的 S。从 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 相邻,且 vi+1 与 S 相邻。证明方法也是根据鸽巢原理,既然与 S 和 T 相邻的点都在该路径上,它们分布的范围只有 v1 ∼ vk 这 k 个点, k ≤ N - 2,而 d(S) + d(T) ≥ N,那么可以想像,肯定存在一个与 S 相邻的点 vi 和一个与 T 相邻的点 vj, 满足 j < i。那么上面的命题也就显然成立了。

找到了满足条件的节点 vi 以后,就可以把原路径变成 S → vi+1 → T → vi → S,即形成了一个回路。
4. 现在我们有了一个没有重复节点的回路。如果它的长度为 N,则汉密尔顿回路就找到了。

如果回路的长度小于 N,由于整个图是连通的,所以在该回路上,一定存在一点与回路以外的点相邻。那么从该点处把回路断开,就变回了一条路径。再按照步骤 1 的方法尽量扩展路径,则一定有新的节点被加进来。接着回到步骤 2。

模板题:POJ 2438 or HDU 4337 Childrens Dining

问题是求小朋友围着桌子的座次就是求原图中的一个环,但是要求这个环不能包含所给出的每条边,所以没给出的边却是可以使用的,也就是说本题实际上是在原图的反图上求一个环,即在每两个可以坐在相邻位置的小朋友连一条边,否则不连。使得该环包含所有顶点,即Hamilton回路。

由于有2n个小朋友,且每个小朋友的敌人最多n-1个,所以,每个小朋友可以一起与座的小朋友最少有n+1个,即度数>=n+1,所以任意两个小朋友度数之和d(u)+d(v)>=2n+2 > 2n,所以Hamilton回路存在。

代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
#define N 407 int vis[N],mp[N][N],ans[N];
int n,m; void init()
{
for(int i=;i<=n;i++)
{
for(int j=;j<=n;j++)
{
if(i == j)
mp[i][j] = ;
else
mp[i][j] = ;
}
}
memset(ans,,sizeof(ans));
}
void reverse(int ans[N],int s,int t) //将ans数组中s到t的部分倒置
{
int tmp;
while(s < t)
{
swap(ans[s],ans[t]);
s++;
t--;
}
} void Hamilton()
{
int s = ,t; //初始化s取1号点
int k = ;
int i,j,w,tmp;
memset(vis,,sizeof(vis));
for(i=;i<=n;i++)
{
if(mp[s][i])
break;
}
t = i; //取任意连接s的点为t
vis[s] = vis[t] = ;
ans[] = s;
ans[] = t;
while()
{
//从t向外扩展
while()
{
for(i=;i<=n;i++)
{
if(mp[t][i] && !vis[i])
{
ans[k++] = i;
vis[i] = ;
t = i;
break;
}
}
if(i > n)
break;
}
//将当前得到的序列倒置,s和t互换,从t继续扩展,相当于在原来的序列上从s扩展
w = k - ;
i = ;
reverse(ans,i,w);
swap(s,t);
//从新的t向外扩展,相当于在原来的序列上从s向外扩展
while()
{
for(i=;i<=n;i++)
{
if(mp[t][i] && !vis[i])
{
ans[k++] = i;
vis[i] = ;
t = i;
break;
}
}
if(i > n)
break;
}
if(!mp[s][t]) //如果s和t不相邻,进行调整
{
for(i=;i<k-;i++)
{
if(mp[ans[i]][t] && mp[s][ans[i+]]) //取序列中一点i,使得ans[i]与t相连接且ans[i+1]与s相连
break;
}
//将从ans[i+1]到t部分的ans[]倒置
w = k - ;
i++;
t = ans[i];
reverse(ans,i,w);
}
//如果当前s和t相连
if(k == n) //如果当前序列中包含n个元素,算法结束
return;
//当前序列中的元素个数小于n,寻找点ans[i],使得ans[i]与ans[]外一点相连
for(j=;j<=n;j++)
{
if(vis[j])
continue;
for(i=;i<k-;i++)
if(mp[ans[i]][j])
break;
if(mp[ans[i]][j])
break;
}
s = ans[i-];
t = j;
reverse(ans,,i-); //将ans[]中s到ans[i-1]部分的ans[]倒置
reverse(ans,i,k-); //将ans[]中ans[i]到t的部分倒置
ans[k++] = j; //将点j加入到ans[]的尾部
vis[j] = ;
}
} int main()
{
int i,j;
int a,b;
while(scanf("%d%d",&n,&m)!=EOF && (n||m))
{
n *= ;
init();
for(i=;i<=m;i++)
{
scanf("%d%d",&a,&b);
mp[a][b] = mp[b][a] = ; //建立反图
}
Hamilton();
printf("%d",ans[]);
for(i=;i<n;i++)
printf(" %d",ans[i]);
printf("\n");
}
return ;
}

Hamilton回路的判定与构造的更多相关文章

  1. Hamilton回路 旅行商TSP问题 /// dp oj1964

    题目大意: 给出一个n个顶点的无向图,请寻找一条从顶点0出发,遍历其余顶点一次且仅一次.最后回到顶点0的回路——即Hamilton回路. Input 多测试用例.每个测试用例: 第一行,两个正整数 n ...

  2. Islands and Bridges(POJ2288+状压dp+Hamilton 回路)

    题目链接:http://poj.org/problem?id=2288 题目: 题意:求Hamilton 路径权值的最大值,且求出有多少条权值这么大的Hamilton路径. 思路:状压dp,dp[i] ...

  3. Noip2016

    <这篇是以前的,不开新的了,借版面来换了个标题> 高二了 开学一周,每天被文化课作业碾压... 但是仍然阻挡不了想刷题的心情... 对付noip2016的几块:(有点少,以后补) 高精度( ...

  4. Java基础常见英语词汇

    Java基础常见英语词汇(共70个) ['ɔbdʒekt] ['ɔ:rientid]导向的                             ['prəʊɡræmɪŋ]编程 OO: object ...

  5. computer English

    算法常用术语中英对照Data Structures 基本数据结构Dictionaries 字典PriorityQueues 堆Graph Data Structures 图Set Data Struc ...

  6. NOIP算法总结

    前言 离NOIP还有一个星期,匆忙的把寒假整理的算法补充完善,看着当时的整理觉得那时还年少.第二页贴了几张从贴吧里找来的图片,看着就很热血的.旁边的同学都劝我不要再放PASCAL啊什么的了,毕竟我们的 ...

  7. 看到了必须要Mark啊,最全的编程中英文词汇对照汇总(里面有好几个版本的,每个版本从a到d的顺序排列)

    java:  第一章: JDK(Java Development Kit) java开发工具包 JVM(Java Virtual Machine) java虚拟机 Javac  编译命令 java   ...

  8. 冲刺NOIP复习,算法知识点总结

    前言        离NOIP还有一个星期,匆忙的把整理的算法补充完善,看着当时的整理觉得那时还年少.第二页贴了几张从贴吧里找来的图片,看着就很热血的.当年来学这个竞赛就是为了兴趣,感受计算机之美的. ...

  9. 专业英语词汇(Java)

    abstract (关键字)             抽象 ['.bstr.kt] access                            vt.访问,存取 ['.kses]‘(n.入口, ...

随机推荐

  1. LINUX重启MYSQL的命令

    LINUX重启MYSQL的命令 标签: mysqllinuxservice脚本web服务server 2010-06-25 10:21 62152人阅读 评论(0) 收藏 举报  分类: Linux( ...

  2. Asp.Net MVC开源论坛中文版

    支持多国语言 支持多种数据库,开盖即饮(因为EF支持),无需安装. 积分 等级 权限 角色 标签 Rss 表情 附件 审核 问答 投票 收藏 日志 排行榜与热点 主题,默认Bootstrap响应式 最 ...

  3. Android数据的四种存储方式SharedPreferences、SQLite、Content Provider和File (一) —— 总览

    Android数据的四种存储方式SharedPreferences.SQLite.Content Provider和File (一) —— 总览   作为一个完成的应用程序,数据存储操作是必不可少的. ...

  4. 初学Node(二)package.json文件

    package.json简介 package.json在Node项目中用于描述项目的一些基本信息,以及依赖的配置,一般每一个Node项目的根目录下都有一个package.json文件. 在项目的根目录 ...

  5. Snort - manual 笔记(二)

    1.5 Packet Acquisition Snort 2.9 引入 DAQ 代替直接调用 libpcap . 有两种网卡特性会影响 Snort : "Large Receive Offl ...

  6. Ubuntu14.04下安装Hadoop2.5.1 (单机模式)

    本文地址:http://www.cnblogs.com/archimedes/p/hadoop-standalone-mode.html,转载请注明源地址. 欢迎关注我的个人博客:www.wuyudo ...

  7. IOS开发之实现App消息推送(最新)

    好久没有写过博客啦,今天就由本菜鸟给大家做一个简单的IOSApp消息推送教程吧!一切从0开始,包括XCode6, IOS8, 以及苹果开发者中心最新如何注册应用,申请证书以及下载配置概要文件,相信很多 ...

  8. 我的mac上的软件(以防优盘重装系统使用)

     

  9. Effective Java 06 Eliminate obsolete object references

    NOTE Nulling out object references should be the exception rather than the norm. Another common sour ...

  10. MFC 网络编程 -- 总结

    原文链接:http://www.cnblogs.com/lidabo/archive/2012/07/19/2598734.html 1.基于 TCP 的 socket 编程 /* 服务器端程序流程: ...