哈密顿问题

基本概念

  1. 哈密尔顿通路:经过图中每个结点且仅经过一次的通路。
  2. 哈密尔顿回路:经过图中每个结点且仅经过一次的回路。
  3. 哈密尔顿图:存在哈密尔顿回路的图。
  4. 竞赛图:每对顶点之间都有一条边相连的有向图,n 个顶点的竞赛图称为 n 阶竞赛图。
  5. 与欧拉回路的对比:欧拉回路是指不重复地走过所有路径的回路;哈密尔顿回路是指不重复地走过所有点并且最后回到起点的回路。
    1.哈密尔顿通路的判定
    设一无向图有 n 个顶点,u、v 为图中任意不相邻的两点,deg(x) 代表 x 的度数
    成立,则图中存在哈密尔顿通路
    2.哈密尔顿回路的判定:Dirac 定理
    设一无向图有 n 个顶点,u、v 为图中任意不相邻的两点,deg(x) 代表 x 的度数
    ,则图中存在哈密尔顿回路
    推论:对于 的无向图,若其任一一点 u 的度数 ,则图中存在哈密尔顿回路
    3.竞赛图
    对于 (点数)的竞赛图,一定存在哈密尔顿通路

状压DP求最短Hamilton

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Z8qLv5xs-1602234310079)(C:\Users\11111\AppData\Roaming\Typora\typora-user-images\image-20201009170245133.png)]

int f[1 << 20][20];
int w[20][20];
int n, m;
int main(){scanf("%d", &n);for(int i = 0; i < n; ++ i)for(int j = 0; j < n; ++ j)scanf("%d", &w[i][j]);memset(f, 0x3f, sizeof f);f[1][0] = 0;for(int i = 0; i <= (1 << n) - 1; ++ i)for(int j = 0; j < n; ++ j){if((i >> j) & 1){for(int k = 0; k < n; ++ k){if(i ^ (1 << j) >> k & 1){f[i][j] = min(f[i][j], f[i ^ (1 << j)][k] + w[k][j]);}}}}printf("%d\n", f[(1 << n) - 1][n - 1]);return 0;
}

dfs 搜索求哈密尔顿回路

以每个点为起点进行搜索,直到形成回路

#define N 101
int n,m;
int u,v;
int g[N][N];
int vis[N],appear[N];
int ans[N],num[N];
int length;
void dfs(int last,int i)//last表示上次访问的点
{  vis[i]=1;//标记为已经访问过  appear[i]=1;//标记为已在一张图中出现过  ans[length++]=i;//记录答案  for(int j=1;j<=num[i];j++)  {  if(g[i][j]==x&&g[i][j]!=last)//回到起点构成哈密顿环  {   ans[++length]=g[i][j];//存储答案for(int i=1;i<=length-1;i++) //找到了一个环,输出anscout<<ans[i]<<' ';  cout<<ans[length]<<endl;length--;//长度-1break;}  if(!vis[g[i][j]])//遍历与i相关联的所有未访问的点。  dfs(i,g[i][j]); }  length--;  vis[i]=0;//回溯
}
int main()
{  memset(vis,0,sizeof(vis));  memset(appear,0,sizeof(appear));  cin>>n>>m; //读入点数与边数for(int i=1;i<=m;i++){   cin>>u>>v; //读入两点g[u][++num[v]]=v;//记录u-v的边g[v][++num[v]]=u;//记录v-u的边}  for(x=1;x<=n;x++) //枚举每一个点,将其作为起点来尝试访问{  if(!appear[x])//如果点x不在之前曾经被访问过的图里  {  length=0;//记录答案的长度  dfs(0,x);  }  }  return 0;
}

Dirac 定理下构造无向图的哈密顿回路

O(n2)O(n^2)O(n2)空间上由于边数非常多,所以采用邻接矩阵来存储比较适合

bool G[N][N];
bool vis[N];
int ans[N];
void Reverse(int arv[N],int s,int t){//将数组anv从下标s到t的部分的顺序反向int temp;while(s<t){swap(arv[s],arv[t]);s++;t--;}
}
void Hamilton(int n){int t;int s=1;//初始化取s为1号点for(int i=1;i<=n;i++)if(G[s][i]){t=i;//取任意邻接与s的点为tbreak;}memset(vis,false,sizeof(vis));vis[s]=true;vis[t]=true;ans[0]=s;ans[1]=t;int ansi=2;while(true){//从t向外扩展while(true){int i;for(i=1;i<=n;i++){if(G[t][i] && !vis[i]){ans[ansi++]=i;vis[i]=true;t=i;break;}}if(i>n)break;}//将当前得到的序列倒置Reverse(ans,0,ansi-1);//s和t互换swap(s,t);while(true){//从t继续扩展,相当于在原来的序列上从s向外扩展int i;for(i=1;i<=n;i++){if(G[t][i] && !vis[i]){ans[ansi++]=i;vis[i]=true;t=i;break;}}if(i>n)break;}//如果s和t不相邻,进行调整if(!G[s][t]){//取序列中的一点i,使得ans[i]与t相连,并且ans[i+1]与s相连int i;for(i=1;i<ansi-2;i++)if(G[ans[i]][t]&&G[s][ans[i+1]])break;i++;t=ans[i];Reverse(ans,i,ansi-1);//将从ans[i+1]到t部分的ans[]倒置}//此时s和t相连//如果当前序列包含n个元素,算法结束if(ansi==n)return;//当前序列中元素的个数小于n,寻找点ans[i],使得ans[i]与ans[]外的一个点相连int i,j;for(j=1;j<=n;j++){if(vis[j])continue;for(i=1;i<ansi-2;i++)if(G[ans[i]][j])break;if(G[ans[i]][j])break;}s=ans[i-1];t=j;//将新找到的点j赋给tReverse(ans,0,i-1);//将ans[]中s到ans[i-1]的部分倒置Reverse(ans,i,ansi-1);//将ans[]中ans[i]到t的部分倒置ans[ansi++]=j;//将点j加入到ans[]尾部vis[j]=true;}
}int main(){int n,m;while(scanf("%d%d",&n,&m)!=EOF&&(n||m)){n*=2;for(int i=0;i<=n;i++){for(int j=0;j<=n;j++){if(i==j){G[i][j]=false;G[j][i]=false;}else{G[i][j]=true;G[j][i]=true;}}}int ansi=0;memset(ans, 0, sizeof(ans));for(int i=1;i<=m;i++){int x,y;scanf("%d%d",&x,&y);G[y][x]=false;G[x][y]=false;}Hamilton(n);for(int i=0;i<n;i++)printf("%d ", ans[i]);printf("\n");}return 0;
}

** N 阶竞赛图下构造有向图的哈密顿通路**

含有N个顶点的有向图,且每对顶点之间都有一条边的图,一定存在哈密顿通路

int ans[105];
int map[105][105];
void Insert(int arv[], int &len, int index, int key){if(index>len)index=len;len++;for(int i=len-1; i>=0; i--){if(i!=index && i)arv[i]=arv[i-1];else{arv[i]=key;return;}}
}
void Hamilton(int n){int ansi = 1;ans[ansi++] = 1;for(int i=2; i<=n; i++){//第一种情况,直接把当前点添加到序列末尾if(map[i][ans[ansi-1]]==1)ans[ansi++]=i;else{int flag=0;//当前序列从后往前找到第一个满足条件的点j,使得存在<Vj,Vi>且<Vi,Vj+1>.for(int j=ansi-2; j>0; j--){if(map[i][ans[j]]==1){//找到后把该点插入到序列的第j+1个点前.flag=1;Insert(ans,ansi,j+1,i);break;}}if(!flag)//否则说明所有点都邻接自点i,则把该点直接插入到序列首端.Insert(ans,ansi,1,i);}}
}
int main(){int n,m;scanf("%d", &n);m=n*(n-1)/2;for(int i=0;i<m;i++){int u,v;scanf("%d%d",&u,&v);if(u<v)map[v][u]=1;}Hamilton(n);for(int i=1;i<=n;i++)printf(i==1? "%d":" %d",ans[i]);printf("\n");return 0;
}

来源

【算法笔记】哈密顿问题相关推荐

  1. 《算法笔记》中文版 - 包括数组,链表,树,图,递归,DP,有序表等相关数据结构与算法的讲解及代码实现...

    来源:专知本文为资源,建议阅读5分钟本文为你分享<算法笔记>中文版. https://github.com/Dairongpeng/algorithm-note 目录概览 第一节 复杂度. ...

  2. 数据结构与算法笔记 - 绪论

    数据结构与算法笔记 - 绪论 1. 什么是计算 2. 评判DSA优劣的参照(直尺) 3. 度量DSA性能的尺度(刻度) 4. DSA的性能度量的方法 5. DSA性能的设计及其优化 x1. 理论模型与 ...

  3. 数据结构与算法笔记(十六)—— 二叉搜索树

    一.二叉搜索树定义 二叉搜索树(Binary Search Tree),又名二叉排序树(Binary Sort Tree). 二叉搜索树是具有有以下性质的二叉树: 若左子树不为空,则左子树上所有节点的 ...

  4. 数据结构与算法笔记(十五)—— 散列(哈希表)

    一.前沿 1.1.直接寻址表 当关键字的全域U比较小时,直接寻址是一种简单而有效的技术.假设某应用要用到一个动态集合,其中每个元素都有一个取自全域U={0,1,-,m-1)的关键字,此处m是一个不很大 ...

  5. 《algorithm-note》算法笔记中文版正式发布!

    无论是做机器学习.深度学习.自然语言处理还是其它领域,算法的重要性不言而喻!吃透算法底层原理.掌握算法数学推导和代码实现,对提高自己的硬核实力来说非常重要!今天给大家推荐一个超赞的开源算法笔记!中文版 ...

  6. 【算法】《algorithm-note》算法笔记中文版正式发布!

    无论是做机器学习.深度学习.自然语言处理还是其它领域,算法的重要性不言而喻!吃透算法底层原理.掌握算法数学推导和代码实现,对提高自己的硬核实力来说非常重要!今天给大家推荐一个超赞的开源算法笔记!中文版 ...

  7. c++ string 删除字符_算法笔记|(5)第二章C、C++的快速入门字符数组的存放方式string.h文件...

    字符数组的存放方式 由于字符数组是由若干个char类型的元素组成的,因此字符数组的每一位都是一个char字符,除此之外,在一维字符数组或者二维字符数组的第二维的末尾都有一个空字符\0表示存放的字符串的 ...

  8. 算法笔记(JavaScript版)——排序

    算法笔记(JavaScript版)--排序 本文内容根据Rebert Sedgewick和Kevin Wayne的<算法(第四版)>整理,原代码为java语言,自己修改为JavaScrip ...

  9. 三维重建7:Visual SLAM算法笔记

    VSLAM研究了几十年,新的东西不是很多,三维重建的VSLAM方法可以用一篇文章总结一下. 此文是一个好的视觉SLAM综述,对视觉SLAM总结比较全面,是SLAM那本书的很好的补充.介绍了基于滤波器的 ...

  10. 最优化理论与算法笔记

    最优化理论与算法笔记

最新文章

  1. SAP MM 特殊库存之T库存
  2. Flume-NG源码阅读之SourceRunner,及选择器selector和拦截器interceptor的执行
  3. 各项技术对正则表达式的支持
  4. 【软考-软件设计师】计算机系统基础知识
  5. make modules 和 make modules_install
  6. react 事件处理_在React中处理事件
  7. Alats2种局部刷新的比较
  8. jdbc防止sql注入-PreparedStatement
  9. EXCEL多元回归分析
  10. 四次考研,终于上岸!反正我感觉很牛逼!
  11. 电脑数据格式化如何快速简单恢复数据?
  12. 一步一步实现一个简单的OS(先小装一下)
  13. win10 icc文件路径
  14. 全国高校大学招标公告信源地址大全分享
  15. Debian 官方下载地址
  16. 清华大学 计算机技术 非全日制,2018年清华大学计算机系计算机技术考研(085211)考试科目、参考书目、复习经验---新祥旭考研...
  17. linux 渗透工具_适用于Linux的十大最佳渗透测试工具
  18. EveryThing下载链接
  19. Python处理音频信号实战 : 手把手教你实现音乐流派分类和特征提取
  20. 支持wifi连接的服务器,WiFi模块连接服务器设置方法

热门文章

  1. 图像处理分类、一般流程与算法
  2. 超赞!YOLOv5的妙用:学习手语,帮助听力障碍群体
  3. Confluence 6 配置文件和key
  4. new File(String Path)加载资源问题
  5. EditText禁止输入回车
  6. 邮件客户端WebMail Pro v7.7.5发布,在线订购限时75折优惠!
  7. 使用ViewBag传送数据从控制器至视图
  8. ASP.NET缓存 Cache之数据缓存
  9. [转|整理]翻译:使用.net3.5的缓存池和SocketAsyncEventArgs类创建socket服务器
  10. java hello world