首页 > 其他分享 >2023-04-04 哈密尔顿问题和路径压缩

2023-04-04 哈密尔顿问题和路径压缩

时间:2023-04-04 23:34:04浏览次数:56  
标签:路径 04 哈密尔顿 2023 访问 回路 visited 顶点

哈密尔顿问题和路径压缩

1 哈密尔顿回路和TSP

路径与回路

哈密尔顿问题偏计算机,欧拉问题偏数学,所以本章我们主要讲哈密尔顿回路和哈密尔顿路径

  • 哈密尔顿回路
  • 哈密尔顿路径
  • 欧拉回路
  • 欧拉路径

哈密尔顿回路定义

从一个点出发,沿着边走,经过每个顶点恰好一次,之后再回到出发点,过程中经过的路径就叫哈密尔顿路径

关键是如下两个特点

  • 回到出发点
  • 经过每个顶点,并且每个顶点只能经过一次

哈密尔顿回路问题的起源

1859年年,爱尔兰数学家哈密尔顿(Hamilton)提出下列列周游世界的游戏:
在正十二⾯面体的二十个顶点上依次标记伦敦、巴黎、莫斯科等世界著名⼤大城市,正十二⾯面体的棱表示连接这些城市的路路线。试问能否在图中做一次旅⾏行行,从顶点到顶点,沿着边⾏行行⾛走,经过每个城市恰好一次之后再回到出发点。这就是著名的哈密尔顿问题
哈密尔顿回路问题的起源
1859年年,爱尔兰数学家哈密尔顿(Hamilton)提出下列列周游世界的游戏。
一个半世纪过去了了,这个问题即⼀一个图是否为哈密尔顿图的判定问题⾄至今悬⽽而未决。数学上:找不不到充分必要条件

哈密尔顿回路问题的变种:旅行推销员问题(Travelling Salesman Problem, TSP)

给定⼀一系列列城市和每对城市之间的距离,求解访问每一座城市⼀次并回到起始城市的最短回路。

  • 带权图,完全图
  • NP难问题(即还没有多项式级别的算法,只有指数级别的算法来解决这类问题)

2 求解哈密尔顿回路的算法

暴力求解

使用排列组合列出所有的顶点组合,遍历所有组合,看能不能找到符合条件的哈密尔顿路径

回溯法:正规解法,比暴力求解性能高一些

回溯算法也叫试探法,它是一种系统地搜索问题的解的方法。回溯算法的基本思想是:从一条路往前走,能进则进,不能进则退回来,换一条路再试。用回溯算法解决问题的一般步骤为:

  • 1、定义一个解空间,它包含问题的解
  • 2、利用适于搜索的方法组织解空间
  • 3、利用深度优先法搜索解空间。
  • 4、利用限界函数避免移动到不可能产生解的子空间。
    问题的解空间通常是在搜索问题的解的过程中动态产生的,这是回溯算法的一个重要特性

下面以下图下图为例说明下回溯法:

举例说明回溯法

注意如下事项:

  • 1.邻接点是TreeSet,所以遍历邻接点是按序号从小到大访问
  • 2.发现某个回路执行完后还有节点没访问,则当前回路不是哈密尔顿回路
  • 3.在2后要进行回退,回退过程中需要把原来访问的节点设置为未访问
  • 4.哈密尔顿回路不允许经过一次顶点两次,所以一旦发现某个顶点的邻接点已经全部被访问,但是图中仍有顶点没被访问,则说明需要回退

详细遍历过程如下:

  • 1.访问0并设置visited[0]=true,0的邻接点有1、2、3,1的序号最小,所以下一个访问1
  • 2.访问1并设置visited[1]=true,1的邻接点有0、2、3,0已经被访问,2和3中2较小,所以下一个访问2
  • 3.访问2并设置visited[2]=true,2的邻接点有0、1,都已经被访问,无法再继续遍历,但是图中顶点3仍然未被遍历,所以需要回退到1
    第1次回退
  • 4.回退到1并设置visited[2]=false,接着访问1剩下的邻接点3(0和2都走过了)
  • 5.访问3并设置visited[3]=true,3的邻接点有0和1,都已经被访问,无法再继续遍历,但是图中顶点2仍然未被访问,所以需要回退到1
  • 6.回退到1并设置visited[3]=false,1的邻接点都已经访问过了仍未找到哈密尔顿回环,接着回退到0
  • 7.回退到0并设置visited[1]=false,0的邻接点1证明走不通,这次走邻接点2
    再次回到0后从2开始访问
  • 8.访问2并设置visited[2]=true,2的邻接点有0、1,0是父节点已经被访问,所以下面访问1
  • 9.访问1并设置visited[1]=true,1的邻接点有0、2、3,0和2都是祖先节点已经被访问,所以下面访问节点3
    这次就能顺利找到哈密尔顿路径了
  • 10.访问3并设置visited[3]=true,此发现3已经和0是邻接点而且图中的点都已经被访问,所以0-2-1-3-0就是哈密尔顿回环
    找到了哈密尔顿回环

注意:回溯时选取一个定点开始就行,不用试遍所有的顶点,因为存在哈密尔顿回路的话,任何一个顶点开始回溯都是可以的;不存在哈密尔顿回路地话,任何一个顶点开始都是找不到回路地

3 求哈密尔顿回路算法(回溯)的实现

graph2.txt的示意图如下,注意此图中回路是从下标1开始,我们自己计算地结果是从0开始
测试图

4 求哈密尔顿回路算法(回溯)的改进

allVisited()方法在所有能到起始点的路径上都会执行一遍,比较损耗性能,我们改成剩余没被访问的顶点数left,未被访问顶点数等于0说明树遍历完毕

5 哈密尔顿路径

首先明确所谓的哈密尔顿路径和哈密尔顿回路两个概念。

比较 定义 代码注意地地方
哈密尔顿路径 由一个起点到达一个终点,经过且只经过图的每个节点一次的路径,起点和终点是不是同一个点都行 同样一张图,从有的点出发,就存在哈密尔顿路径,从另一个点出发,就不存在哈密尔顿路径。所以,我们在算法设计中,构造函数需要用户显式地传入起始点
哈密尔顿回路 是一个闭合的哈密尔顿路径,即:起点和终点是同一个点 求哈密尔顿回路,选取不同的起始点对结果无影响,如果存在回路,那么图所有的点都是回路上的点,选任何一个顶点进行DFS回路检测效果都是一样的,所以在构造函数里随便传入一个起始点即可,一般选0

6 哈密尔顿路径应用举例

LeetCode980号问题 https://leetcode-cn.com/problems/unique-paths-iii/submissions/

7 状态压缩

前面用到的状态压缩

  • 水桶倒水问题,两个水桶的水量用一个两位数表示(v=a*10+b)
  • 栅格问题,二维的坐标用一维表示(v=r*C+c)
  • ...

状态压缩常用操作

以DFS中visited数组为例,visited一个二进制数来表示

  • visited数组用一个二进制数表示

    visited数组用一个二进制数表示

  • 判断visited的第i位是否为1

    取出visited数组的第i位
    通用化地提取出公式如下:
    取出visited数组的第i位2

  • 设置visited的第i位为1

    设置visited的第i位为1

  • 设置visited的第i位为0

    设置visited的第i位为0

综上,状态压缩常见操作的公式如下:

操作 公式
判断第i位是否为1 visited & (1<<i) != 0
如果第i为为0,设为1 visited + (1<<i)
如果第i为为1,设为0 visited - (1<<i)

状态压缩在哈密尔顿路径中的实践:
状态压缩在哈密尔顿路径中的实践

8 基于状态压缩的哈密尔顿算法

自己改造下上面的哈密尔顿回路、哈密尔顿路径以及LeetCode980号问题的实现

9 记忆化搜索

  • 当有大量重复数据的时候适合用回溯法
  • 当数据量很大同时重复数据不多时,O(n*2^n)的性能要比O(n!)很多,此时适合用记忆化搜索
    记忆化搜索

10 本章小节

本章小节

标签:路径,04,哈密尔顿,2023,访问,回路,visited,顶点
From: https://www.cnblogs.com/lsgwr/p/17288267.html

相关文章

  • java学习日记20230406-StringBuilder,StringBuffer,String比较
    StringBuffer,StringBuilder,String比较: StringBuilder和StringBuffer非常类似,均代表可变的字符序列,而且方法相同;String:不可变字符序列,效率低,但是复用率高;StringBuffer:可变字符序列,效率较高,线程安全;StringBuider:可变字符序列,效率极高,线程不安全  String使用注意说明: ......
  • java学习日记20230406-StringBuilder类
    StringBuilder类一个可变的字符序列,此类提供一个与StringBuffer兼容的Api,但不保证同步。该类被设计用作StringBuffer的一个简易替换,用在字符串缓冲区被单个线程使用的时候。如果可能,建议优先采用该类,因为在大多数实现中,他比StringBuffer要快----StringBuilder不是线程安全的在S......
  • ZSTU2023校赛
    篠塚真佑実的树给定\(n\)个节点的树,其中\(m\)个节点存在传送门,当飞船经过存在传送门的节点的时候,可以选择无消耗地传送至其他存在传送门的节点,现在有\(q\)次询问,每次询问给出起点\(st\)和终点\(ed\),若每艘飞船在飞行中最多只能进行一次传送,请你输出每次询问从起点到终点的最短路......
  • C/C++物业费管理系统[2023-04-04]
    C/C++物业费管理系统[2023-04-04]程序设计题:物业费管理系统出题人:俞琼面向专业:软件工程难度等级:41问题描述为维护小区正常的运营管理,居民应按时缴纳小区管理费,请设计一个物业费管理系统,对小区的住户进行收费管理。通过此课题,熟练掌握文件、数组、指针的各种操作,以及一些基......
  • 每日总结2023/3.28(pycharm创建pp工程)
            defcalculate_fee(distance_travelled):return10+2*distance_travelledforxin[1.0,3.0,5.0,9.0,10.0,20.0]:print(calculate_fee(x))   ......
  • java学习日记20230405-StringBuffer类
    StringBuffer类java.lang.StringBuffer代表可变的字符序列,可以对字符串内容进行增删很多方法与String相同,但StringBuffer是可变长度的StringBuffer是一个容器StringBuffer是final类实现了Serializable接口,可以保存到文件或网络传输继承了抽象类AbstractStringBuiderAbstra......
  • MATLA 5G工具箱---2023小迈步之通信系统设计——从基础到 AI+
    基于MATLABR2022b版软件学习!【官方】2023小迈步之通信系统设计——从基础到AI+(上)_哔哩哔哩_bilibili  ImproveSNRandCapacityofWirelessCommunicationUsingAntennaArrays利用天线阵列提高无线通信的信噪比和容量Thegoalofawirelesscommunicat......
  • 工作感受月记(202304月)
    2023年04月04日记录自己小手术后的第二天上班日状态。因为又不能低埋,也不能太仰视,以避免伤口被拉开。在此状态下,自己工作不能太专心,可以说是靠意念来完成一天的工作。今日工作事项:1/昨日的一个task,查看armtemplate的parametertype为int时候,minvlue和maxvalue值不能有效......
  • NOI2023联合省选游记
    前言这次比赛去了广大附中,也是我初中最后一场比赛了吧Day-DY的由于里的太近了,住在学校里,感到惋惜。训练量有点少,不知道因该干什么,于是随便看了几个简单数学知识。譬如拉格朗日,三维计算几何。比较冷门,考试没有太大用途,只能赌一把,看看它考不考。Day1第一天早上6点20起来,酒店的......
  • 4.04每日总结
    以下是SQLSELECT语句使用WHERE子句从数据表中读取数据的通用语法:SELECTfield1,field2,...fieldNFROMtable_name1,table_name2...[WHEREcondition1[AND[OR]]condition2.....查询语句中你可以使用一个或者多个表,表之间使用逗号, 分割,并使用WHERE语句来设定查询条......