AcWing 731. 毕业旅行问题(每日一题)

2024-04-05 01:20

本文主要是介绍AcWing 731. 毕业旅行问题(每日一题),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原题链接:731. 毕业旅行问题 - AcWing题库

此题难度较大,是2019年字节跳动校招题,里面涉及位运算与状态压缩DP,不会的可以去学习,此题根据个人量力而行。

建议看一下y总的讲解:AcWing 731. 毕业旅行问题(每日一题)_哔哩哔哩_bilibili

目录

题目:

解题思路:

代码实现:

写在最后:


题目:

小明目前在做一份毕业旅行的规划。

打算从北京出发,分别去若干个城市,然后再回到北京,每个城市之间均乘坐高铁,且每个城市只去一次。

由于经费有限,小明希望能够通过合理的路线安排尽可能的省些路上的花销。

给定一组城市和每对城市之间的火车票的价钱,找到每个城市只访问一次并返回起点的最小车费花销。

注意北京为 1 号城市

输入格式

第一行包含一个正整数 n,表示城市个数。

接下来输入一个 n 行 n列的矩阵,表示城市间的车票价钱。

输出格式

输出一个整数,表示最小车费花销。

数据范围

1<n≤20,包括北京
车票价格均不超过 1000 元。

输入样例:

4
0 2 6 5
2 0 4 4
6 4 0 2
5 4 2 0

输出样例:

13

说明

共 4 个城市,城市 1 和城市 1 的车费为 0,城市 1 和城市 2 之间的车费为 2,城市 1 和城市 3 之间的车费为 6,城市 1 和城市 4 之间的车费为 5,以此类推。

假设任意两个城市之间均有单程票可买,且价格均在 1000 元以内,无需考虑极端情况。


解题思路:

此题主要用了状态压缩DP,它的f[i,j]设置的很巧妙,把i化成二进制,比如有五个城市,初始为00001,最后一位为北京当前就在北京。下一个比如武汉花费小(武汉第三个城市)那就是二进制为00101,又到达上海(第五个城市)那就是10101……当达到目标状态二进制为11111,把所有城市都标记一遍,那么此次完成了此次旅行,还一个问题那就是我还要回北京啊,既然要回北京那就要找最后一个到达的城市,从最后到达的城市坐高铁直接回北京,那么我们最后还要枚举一遍最后到达的城市,在所有城市里面加上回北京的费用,取一个最小值即是答案。

一些细节问题以及一些位运算等问题以图片手写解答一下:


代码实现:

#include<iostream>
#include<cstring>
using namespace std;
const int N=20,M=1<<N,inf=0x3f3f3f;
int n;
int w[N][N],f[M][N];//w[i][j]表示花费数组
//f[i][j]表示i为二进制标记数组,j表示到达j个城市的最小花费
int main(){cin>>n;for(int i=0;i<n;i++){for(int j=0;j<n;j++){cin>>w[i][j];}}memset(f,inf,sizeof(f));//初始无穷大f[1][0]=0;//2^0=1第一个北京地点被标记为1for(int i=1;i<1<<n;i+=2){//1~2^n都遍历一遍//i+2是把第一个北京也给同时被标记,加一二进制逢二进一,会变成10,北京又被置成0没有被标记,所以加2还是1for(int j=0;j<n;j++){//j遍历到达第j个城市if(i>>j&1){//此时处于j城市,看看是否i中包含j(被标记),i的二进制表示第j位是否为1for(int k=0;k<n;k++){//k表示中转城市到达j城市if(i-(1<<j)>>k&1){//判断第i个状态除去第j个城市是否包含第k个城市f[i][j]=min(f[i][j],f[i-(1<<j)][k]+w[k][j]);}}}}}int res=inf;for(int i=1;i<n;i++){//枚举最后一个到达的城市res=min(res,f[(1<<n)-1][i]+w[i][0]);//为了二进制方便下标从0开始}cout<<res<<endl;return 0;
}

写在最后:

此题对于博主这样的cj来说比较难了,都是按照博主自己的理解去写的,会有一些不是很准确的地方,学了很长时间,似懂非懂的感觉,大佬勿喷。主要还是状态的定义利用二进制去标记那个地方第一次见,感觉很巧妙,再就是写代码中不经常使用位运算对于y总的代码看起来也比较困难,还有很多地方需要好好学习一次,感觉跟着y总的每日一题学到了不少东西,以后还需继续加油,文章若有错误的地方请各位大佬指出,一起学习进步。

PS:第一张图片来组y总讲解  作者:yxc

这篇关于AcWing 731. 毕业旅行问题(每日一题)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/877313

相关文章

MySQL 设置AUTO_INCREMENT 无效的问题解决

《MySQL设置AUTO_INCREMENT无效的问题解决》本文主要介绍了MySQL设置AUTO_INCREMENT无效的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录快速设置mysql的auto_increment参数一、修改 AUTO_INCREMENT 的值。

关于跨域无效的问题及解决(java后端方案)

《关于跨域无效的问题及解决(java后端方案)》:本文主要介绍关于跨域无效的问题及解决(java后端方案),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录通用后端跨域方法1、@CrossOrigin 注解2、springboot2.0 实现WebMvcConfig

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Java死锁问题解决方案及示例详解

《Java死锁问题解决方案及示例详解》死锁是指两个或多个线程因争夺资源而相互等待,导致所有线程都无法继续执行的一种状态,本文给大家详细介绍了Java死锁问题解决方案详解及实践样例,需要的朋友可以参考下... 目录1、简述死锁的四个必要条件:2、死锁示例代码3、如何检测死锁?3.1 使用 jstack3.2

解决JSONField、JsonProperty不生效的问题

《解决JSONField、JsonProperty不生效的问题》:本文主要介绍解决JSONField、JsonProperty不生效的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录jsONField、JsonProperty不生效javascript问题排查总结JSONField

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三

MySQL版本问题导致项目无法启动问题的解决方案

《MySQL版本问题导致项目无法启动问题的解决方案》本文记录了一次因MySQL版本不一致导致项目启动失败的经历,详细解析了连接错误的原因,并提供了两种解决方案:调整连接字符串禁用SSL或统一MySQL... 目录本地项目启动报错报错原因:解决方案第一个:第二种:容器启动mysql的坑两种修改时区的方法:本地

springboot加载不到nacos配置中心的配置问题处理

《springboot加载不到nacos配置中心的配置问题处理》:本文主要介绍springboot加载不到nacos配置中心的配置问题处理,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录springboot加载不到nacos配置中心的配置两种可能Spring Boot 版本Nacos

Java中JSON格式反序列化为Map且保证存取顺序一致的问题

《Java中JSON格式反序列化为Map且保证存取顺序一致的问题》:本文主要介绍Java中JSON格式反序列化为Map且保证存取顺序一致的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未... 目录背景问题解决方法总结背景做项目涉及两个微服务之间传数据时,需要提供方将Map类型的数据序列化为co

如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socket read timed out的问题

《如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socketreadtimedout的问题》:本文主要介绍解决Druid线程... 目录异常信息触发场景找到版本发布更新的说明从版本更新信息可以看到该默认逻辑已经去除总结异常信息触发场景复