c++ 旅行商问题(动态规划)

目录

  • 一、旅行商问题简介
    • 旅行商问题
    • 问题概述
    • 问题由来
  • 二、基本思路
  • 三、实现
    • 1、状态压缩
    • 2、状态转移
  • 四、代码
  • 五、复杂度分析

一、旅行商问题简介

旅行商问题

  TSP,即旅行商问题,又称TSP问题(Traveling Salesman
Problem),是数学领域中著名问题之一。

问题概述

  假设有一个旅行商人要拜访N个城市,他必须选择所要走的路径,路径的限制是每个城市只能拜访一次,而且最后要回到原来出发的城市。路径的选择目标是要求得的路径路程为所有路径之中的最小值。TSP问题是一个NPC问题。

问题由来

  TSP的历史很久,最早的描述是1759年欧拉研究的骑士周游问题,即对于国际象棋棋盘中的64个方格,走访64个方格一次且仅一次,并且最终返回到起始点。

  TSP由美国RAND公司于1948年引入,该公司的声誉以及线形规划这一新方法的出现使得TSP成为一个知名且流行的问题。

示例:
在这里插入图片描述

黑色数字代表点、红色代表路径的花费

输入:

4 6
1 2 1
1 4 2
1 3 4
2 3 1
2 4 2
3 4 3

输出:

运行中...最短距离为:72条最短路径:
路径11-->4-->3-->2-->1
路径21-->2-->3-->4-->1

提示:

第一行输入点的个数n和边的个数m,点的编号为1~n
接下来m行输入m条边以及花费,p1 p2 v,表示点p1和点p2之间有一条无向边,边的花费为v

二、基本思路

  1、我们需要知道,我们求的路径是一个环,所以无论从哪里开始,结果都应该是一样的,就像例题中的;
最短路径可表示为 1–>4–>3–>2–>1
那么它也可以表示为 4–>3–>2–>1–>4
还可以表示为 3–>2–>1–>4–>3

所以我们可以从任意的点出发去查找路径

  2、旅行商问题只有当图是哈密顿图时才可能有解的,即需要满足题意,可以从一个点出发,到达所有的点一次,然后回到起点。

这个可以通过最后运行的结果判断,我们令初始答案是一个很大的值,如果查找后答案没有被改变,则该图无解

  3、按照传统的暴力搜索,时间复杂度为O(n!),而动态规划可以将复杂度减低到O(n2*2n)

  4、有个注意的点,起点需要走两遍,为了简化问题,只需要预处理从起点走到其它点的最小花费,而起点不能标记为已经走过,因为后面还有回到原点

三、实现

1、状态压缩

  我们需要表达我们已经走过了哪些点,目前到达了哪里,有什么办法表达出来呢?

  暴力是万能的,我们可以开一个数组dp[i][j],代表目前到达了i点,dp[i][j]的值代表j点是否已经走过了,但是这样做的话我们状态转移会变得很麻烦,状态压缩就是它的优化

  状态压缩是通过二进制实现的,我们知道int有32位,那么我们可以用第0位代表第0个点的状态,第1位代表第1个点状态…第n位代表第n个点的状态,位的值如果是1的话就代表该点已经走过了,例如17的二进制为0000010001,代表第0个点和第4个点已经走过了

  那么我们可以开一个数组dp[i][j],代表目前走到了i点,用j代表已经哪些点,例如:
dp[0][17],17的二进制为0000010001,代表目前在第0个点,已经走过第0个点和第4个点。
dp[16][17],17的二进制为0000010001,代表目前在第4个点,已经走过第0个点和第4个点。

  我们可以用dp[i][j]的值代表当前这个状态的最小花费,例如dp[0][17]=12,那么就代表到达该状态需要的最小花费是12

2、状态转移

  dp的基本思想就是记录某个状态的最优解,再从目前的状态转移到新的状态,从局部最优解转移到全局最优解

  我们用数组a[i][j]存储图,那么a[i][j]的值就代表从i点到j点的花费

  我们如何求状态dp[0][19]的最优解?
  19的二进制是0000010011,因为18的二进制为0000010010,那么dp[0][19]可以由dp[4][18],dp[1][18]转移过来,最小花费是dp[0][19]=min(dp[4][18]+a[4][0],dp[1][18]+a[1][0])

  即我们要求大的状态,那么就需要先把小状态最优解求出来。反过来我们求出了所有小状态,那么就可以求出大状态的最优解

在这里插入图片描述

  {1,2,3}代表第1、2、3个点都已经走过了

  可以发现,小状态总是比大状态小的,那么我们可以从0状态枚举到2n-1状态,获取到每个状态的最优解

我们还可以反过来想,从小状态去更新大的状态
在这里插入图片描述
  两种思路都可以

四、代码

  下面代码是基于逆向思想的,即从小状态更新大状态。理解透了的同学不妨尝试写一下大状态调用小状态更新的代码

#include<bits/stdc++.h>
using namespace std;
int n,m;//n点的个数,m边的个数 
int a[15][15];//邻接矩阵存无向图 
int dp[15][1<<15];//dp[i][j]代表从最后走到i点到达状态j 
int t;//一共有t个状态 void init(){//初始化 memset(a,0x3f,sizeof a);memset(dp,0x3f,sizeof dp);cout<<"请输入点和边的个数:"<<endl;cin>>n>>m;cout<<"请输入"<<m<<"条边:"<<endl;for(int i=0;i<m;i++){int x,y,val; cin>>x>>y>>val;x--;y--;a[x][y]=val;a[y][x]=val;}
} void run(){//dp核心算法 t=(1<<n);for(int i=1;i<n;i++){//因为起点初始不能被标记已经走过,所以需要手动初始化起点到达其它点的花费 dp[i][1<<i]=a[0][i];}for(int i=0;i<t;i++){//枚举每一个状态 for(int j=0;j<n;j++){//枚举每一个没有走过的点 if(((i>>j)&1)==0){for(int k=0;k<n;k++){//枚举每一个走过的点 if(((i>>k)&1)==1&&dp[j][i^(1<<j)]>dp[k][i]+a[k][j]){//取最优状态 dp[j][i^(1<<j)]=dp[k][i]+a[k][j];}}}}}
}int tt;//记录 
vector<int> path(1,0);//初始化从0点出发 ,存储单条路径 
vector<vector<int> > paths;//存储所有的路径 
void getPath(int p){//递归查找所有路径 if((tt^(1<<p))==0){//如果是最后一个点了就存储改路径 paths.push_back(path);return; }for(int j=1;j<n;j++){//回溯算法,一个加法的原则//如果点1到达点5的最短距离为100,点1到达点3的最短距离是70//而点3和点5之间的距离为30 ,那么点3是点1到5之间的一个中间点//即1-->...-->3-->5 if(a[j][p]+dp[j][tt^(1<<p)]==dp[p][tt]){tt^=(1<<p);path.push_back(j);getPath(j);tt^=(1<<p);path.pop_back();}}} void print(){//打印路径 cout<<"最短距离为:"<<dp[0][t-1]<<endl;cout<<"共"<<paths.size()<<"条最短路径:" <<endl; for(int i=0;i<paths.size();i++){cout<<"路径"<<i+1<<":1";for(int j=paths[i].size()-1;j>=0;j--){cout<<"-->"<<paths[i][j]+1;}cout<<endl;}
}int main(){init();cout<<"运行中..."<<endl<<endl; run(); cout<<"运行结果:"<<endl; if(dp[0][t-1]==0x3f3f3f3f){//无解 cout<<"该图不是哈密顿图!"<<endl;return 0;}tt=t-1;getPath(0);print(); 
} 

五、复杂度分析

  时间复杂度: 求最小花费枚举2n种状态,每种状态枚举每一个没有走过的点,每一个没走过的点需要枚举每一个已经走过的点,时间复杂度O(n2*2n),求所有路径,时间复杂度将退化为O(n!)
  空间复杂度: 记录每个点的2n种状态,空间复杂度O(n*2n)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/28855.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ChatGPT 最佳实践指南之:系统地测试变化

Test changes systematically 系统地测试变化 Improving performance is easier if you can measure it. In some cases a modification to a prompt will achieve better performance on a few isolated examples but lead to worse overall performance on a more representa…

医疗健康大数据:应用实例与系统分析

来源&#xff1a;网络大数据 1 、概述 随着信息技术和物联网技术的发展、个人电脑和智能手机的普及以及社交网络的兴起&#xff0c;人类活动产生的数据正以惊人的速度增长。根据国际数据公司(International DataCorporation&#xff0c;IDC)的报告&#xff0c;仅2011年&#xf…

夏季达沃斯论坛《2023年十大新兴技术报告》正式公布

来源&#xff1a;悦智网 该报告概述了未来3-5年内有望对社会产生积极影响的技术。该报告的范围不仅仅描述了技术及其相关的风险和机遇&#xff0c;还包括了对每项技术如何对人类、地球、繁荣、产业和公平产生影响的定性评估。 在夏季达沃斯论坛&#xff08;世界经济论坛第十四届…

音视频技术开发周刊 | 292

每周一期&#xff0c;纵览音视频技术领域的干货。 新闻投稿&#xff1a;contributelivevideostack.com。 谷歌将 AI 芯片团队并入云计算部门 追赶微软和亚马逊 OpenAI推出的ChatGPT获得一定成功&#xff0c;微软是OpenAI的重要投资者&#xff0c;它将ChatGPT植入必应搜索&#…

CollovGPT——人工智能工具颠覆传统室内设计行业

作为线上室内设计领先的平台&#xff0c;Collov一直致力于使用先进的技术重新定义「室内设计」&#xff1a;让室内设计不再是一种奢侈品&#xff0c;而是每一个人都可以享受的生活体验。 经过两年的迭代和开发&#xff0c;我们现在正式上线CollovGPT — 一款基于Stable Diffusi…

扩散模型和Transformer梦幻联动!一举拿下新SOTA

作者丨羿阁 萧箫 来源丨量子位 导读 “U-Net已死&#xff0c;Transformer成为扩散模型新SOTA了&#xff01;” 就在ChatGPT占尽AI圈风头时&#xff0c;纽约大学谢赛宁的图像生成模型新论文横空出世&#xff0c;收获一众同行惊讶的声音。 MILA在读ML博士生Ethan Caballero 论文…

92K Star !AI 都完全不需要咱们人类了?

Auto-GPT 究竟是一个开创性的项目&#xff0c;还是一个被过度炒作的 AI 实验&#xff1f;本文为我们揭开了喧嚣背后的真相&#xff0c;并揭示了 Auto-GPT 不适合实际应用的生产局限性。 作者&#xff1a;Jina AI 创始人兼 CEO 肖涵博士 译者&#xff1a; 新智元编辑部 原文链接…

揭秘 Auto-GPT 喧嚣背后的残酷真相!

Auto-GPT 究竟是一个开创性的项目&#xff0c;还是一个被过度炒作的 AI 实验&#xff1f;本文为我们揭开了喧嚣背后的真相&#xff0c;并揭示了 Auto-GPT 不适合实际应用的生产局限性。 本文来自 Jina 官方投稿&#xff0c;作者为 Jina AI 创始人兼 CEO 肖涵博士&#xff0c;如…

通过ChatGPT使用Mermaid.js生成时间序列图、组织结构图等

1、用mermaid.js 生成京东网站改版时间序列图 以下是使用Mermaid.js生成的京东网站改版时间序列图&#xff1a; gantttitle 京东网站改版时间序列图dateFormat YYYY-MM-DDsection 基础功能改版登录注册界面 :done, 2018-01-15, 10d购物车页面优化 :done, 2018-02-10, 10d商…

淘汰ChatGPT的Auto-GPT是炒作?自己跑代码,不需要人类,GitHub已破5万星

视学算法报道 编辑&#xff1a;编辑部 【导读】Auto-GPT究竟是一个开创性的项目&#xff0c;还是一个被过度炒作的AI实验&#xff1f;这篇文章为我们揭开了喧嚣背后的真相&#xff0c;并揭示了Auto-GPT不适合实际应用的局限性。 这两天&#xff0c;Auto-GPT——一款让最强语言…

AIPRM for ChatGPT 提示词模板扩展工具实践

&#xff08;1&#xff09;基本介绍 AIPRM for ChatGPT是一个Chrome浏览器扩展程序&#xff0c;基于Chromium内核开发的浏览器都可以使用该扩展&#xff0c;比如微软的Edge浏览器等。 在AIPRM的帮助下&#xff0c;我们可以在ChatGPT中一键使用各种专门为网站SEO、SaaS、营销、…

惊!掌握通义千问的关键,从这些必知内容开始!

今年快过半了&#xff0c;要说顶流话题还得是ChatGPT&#xff0c;相关话题的热度居高不下&#xff0c;而其从GPT-3.5到GPT-4的升级&#xff0c;也让我们深刻了解了什么叫一代版本一代神&#xff0c;从GPT-3.5到GPT-4&#xff0c;真的就是一个跨阶级式的升级。 技术内涵 ChatGPT…

讯飞星火大模型申请及测试:诚意满满

“ 大家好&#xff0c;我是可夫小子&#xff0c;关注AIGC、读书和自媒体。解锁更多ChatGPT、AI绘画玩法。加&#xff1a;keeepdance&#xff0c;备注&#xff1a;chatgpt&#xff0c;拉你进群。 最近国产大模型跟下饺子似&#xff0c;隔几天就发布一个。厂家发布得起劲&#xf…

拍摄电话?窃听邮件?了解社会工程学攻击和你可能受到的风险

数据来源 本文仅用于信息安全的学习&#xff0c;请遵守相关法律法规&#xff0c;严禁用于非法途径。若观众因此作出任何危害网络安全的行为&#xff0c;后果自负&#xff0c;与本人无关。 社会工程学 社会工程学-渗透测试 社会工程学作用 亦思社会工程学 你注册过哪些网站&…

文心千帆为你而来

1. 前言 3月16号百度率先发布了国内第一个人工智能大语言模型—文心一言。文心一言的发布在业界引起了不小的震动。而文心一言的企业服务则由文心千帆大模型平台提供。文心千帆大模型平台是百度智能云打造出来的一站式大模型开发与应用平台&#xff0c;提供包括文心一言在内的…

第二弹进阶吴恩达 ChatGPT Prompt 技巧

第一弹笔记在这里&#xff1a; 总结吴恩达 ChatGPT Prompt 免费课程 今天分享第二弹&#xff0c;进阶篇。 第一点&#xff0c;任务序列化。 通常看完一篇长文&#xff0c;脑子里往往充满无数疑问。急切想知道所有答案&#xff0c;必须列一个问题清单。对话式问法&#xff0c;对…

CVPR2023论文速递(2023.3.22)!已接入ChatGPT总结!共31篇!

整理&#xff1a;AI算法与图像处理 CVPR2023论文和代码整理&#xff1a;https://github.com/DWCTOD/CVPR2023-Papers-with-Code-Demo 欢迎关注公众号 AI算法与图像处理&#xff0c;获取更多干货&#xff1a; 大家好, 最近正在优化每周分享的CVPR论文, 目前考虑按照不同类别去分…

Python与ChatGPT

Python的用途非常广泛&#xff0c;很多应用场景都可以使用 python 来满足自己的需求&#xff0c;比如自己平常使用 Python 来做网络应用后端开发、做批量处理小工具、做测试软件等&#xff0c;而目前非常热门的 ChatGPT 也与 python 有很大的关系。 据了解&#xff0c;在ChatG…

IOS越狱---checkra1n windows系统越狱

本篇教程适用小白初次越狱&#xff0c;无高阶操作&#xff0c;大佬请止步&#xff0c;本篇教程可能没有任何能学习的地方&#xff0c;以下问题如有不清楚的地方欢迎加微信 vaintech讨论交流 首先介绍所需要的工具 一支U盘&#xff08;2g以上&#xff09;一台电脑要被越狱的手…

【iOS逆向】某App越狱检测

1.目标 此篇文本为入门文章&#xff0c;大家莫抱过多期望。此文章的目的是教大家如何从UI入手&#xff0c;去定位自己想要的东西。 2.操作环境 mac系统 frida-ios-dump&#xff1a;砸壳 已越狱iOS设备&#xff1a;脱壳及frida调试 IDA Pro&#xff1a;静态分析 3.流程 …