题目描述

某乡有n个村庄(1

输入格式:

村庄数n和各村之间的路程(均是整数)。

输出格式:

最短的路程。

输入样例:

3

0 2 1

1 0 2

2 1 0

输出样例

3

说明

输入解释

3 {村庄数}

0 2 1 {村庄1到各村的路程}

1 0 2 {村庄2到各村的路程}

2 1 0 {村庄3到各村的路程}


题目分析

又是一道状压DP经典

我们以一串二进制数表示村庄的集合(状态)

1表示该村庄访问过,0表示没有

定义dp[i][j]表示

从起点到第j号点

且到达时状态恰好为i的最短路

则最后答案就是min(dp[(1<< n) -1][i] + map[i][1]) (2<=i<=n)

其中map数组是题目给定的各村庄间距离

而求取dp数组的方法

我们可以借鉴Floyd的思想

具体代码:

  1. for(int i =0;i<=(1 << n) -1;i++)
  2. for(int j=1;j<=n;j++)
  3. if( !( (1 << j-1) & i) )
  4. for(int k=1;k<=n;k++)
  5. if( ( (1 << k-1) & i) )
  6. dp[((1 << j-1) | i)][j] = min(dp[((1 << j-1) | i)][j],dp[i][k] + map[k][j]);

如何解释呢

第一层循环 i 枚举每个可能的状态

第二层循环 j 枚举下一步到达的点

if( !( (1 << j-1) & i) ) 这句判断 j 是否已访问

1左移j-1位,则此时只有第j位是1

若状态 i 的第 j 位为1,则&与运算返回1,表示已访问,否则没访问

第三层循环枚举中介点k

if语句判断同上

接下来核心代码——更新

dp[ (1 << j-1) | i ][j] = min(dp[((1 << j-1) | i)][j],dp[i][k] + map[k][j]);

(1 << j-1) | i 将状态的第j为置为1得到下一步的状态

dp[i][k] + map[k][j])表示在当前状态i中寻找中介点检查最短路是否可以更新


  1. #include<iostream>
  2. #include<cstdio>
  3. #include<vector>
  4. #include<algorithm>
  5. #include<queue>
  6. #include<cstring>
  7. using namespace std;
  8. int read()
  9. {
  10. int f=1,x=0;
  11. char ss=getchar();
  12. while(ss<'0'||ss>'9'){if(ss=='-')f=-1;ss=getchar();}
  13. while(ss>='0'&&ss<='9'){x=x*10+ss-'0';ss=getchar();}
  14. return f*x;
  15. }
  16. int n;
  17. int dp[1100010][25];
  18. int map[25][25];
  19. int main()
  20. {
  21. n=read();
  22. for(int i = 1;i <= n;i ++)
  23. for(int j = 1;j <= n;j ++)
  24. map[i][j]=read();//读入个村庄间距离
  25. memset(dp,63,sizeof(dp));
  26. dp[1][1] = 0;//状态1表示此时只有1号点访问过
  27. for(int i =0;i<=(1 << n) -1;i++)
  28. for(int j=1;j<=n;j++)
  29. if( !( (1 << j-1) & i) )
  30. for(int k=1;k<=n;k++)
  31. if( ( (1 << k-1) & i) )
  32. dp[((1 << j-1) | i)][j] = min(dp[((1 << j-1) | i)][j],dp[i][k] + map[k][j]);//核心代码,解释如上所述
  33. int ans = 2147483640;
  34. for(int i=2;i<=n;i++)//最后从状态(1<<n)-1(二进制全为1)中寻找到1最短的点
  35. ans=min(ans,dp[(1<<n)-1][i] + map[i][1]);
  36. cout<<ans;
  37. return 0;
  38. }

蒟蒻看到这题n<=20的范围挺大

担心卡常所以开了O2

最后总共跑了1392ms,最大的点跑了608ms

目测不开O2也是能过的

洛谷P1171 售货员的难题【状压DP】的更多相关文章

  1. 2018.07.18 洛谷P1171 售货员的难题(状压dp)

    传送门 感觉是一道经典的状压dp,随便写了一发卡了卡常数开了个O(2)" role="presentation" style="position: relati ...

  2. 洛谷P1171 售货员的难题

    P1171 售货员的难题 题目背景 数据有更改 题目描述 某乡有n个村庄(1<n<20),有一个售货员,他要到各个村庄去售货,各村庄之间的路程s(0<s<1000)是已知的,且 ...

  3. 洛谷 P1171 售货员的难题

    P1171 售货员的难题 题目背景 数据有更改 题目描述 某乡有n个村庄(1<n<20),有一个售货员,他要到各个村庄去售货,各村庄之间的路程s(0<s<1000)是已知的,且 ...

  4. 【题解】洛谷P3959 [NOIP2017TG] 宝藏(状压DP+DFS)

    洛谷P3959:https://www.luogu.org/problemnew/show/P3959 前言 NOIP2017时还很弱(现在也很弱 看出来是DP 但是并不会状压DP 现在看来思路并不复 ...

  5. codevs2596 售货员的难题(状压dp)

    2596 售货员的难题  时间限制: 1 s  空间限制: 32000 KB  题目等级 : 钻石 Diamond     题目描述 Description 某乡有n个村庄(1<n<=15 ...

  6. 洛谷 P1278 单词游戏 【状压dp】

    题目描述 Io和Ao在玩一个单词游戏. 他们轮流说出一个仅包含元音字母的单词,并且后一个单词的第一个字母必须与前一个单词的最后一个字母一致. 游戏可以从任何一个单词开始. 任何单词禁止说两遍,游戏中只 ...

  7. 洛谷P2704 [NOI2001]炮兵阵地 [状压DP]

    题目传送门 炮兵阵地 题目描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图 ...

  8. 洛谷P2761 软件补丁问题(状压DP,SPFA)

    题意 描述不清... Sol 网络流24题里面怎么会有状压dp?? 真是狗血,不过还是简单吧. 直接用$f[sta]$表示当前状态为$sta$时的最小花费 转移的时候枚举一下哪一个补丁可以搞这个状态 ...

  9. 洛谷P2831 愤怒的小鸟——贪心?状压DP

    题目:https://www.luogu.org/problemnew/show/P2831 一开始想 n^3 贪心来着: 先按 x 排个序,那么第一个不就一定要打了么? 在枚举后面某一个,和它形成一 ...

随机推荐

  1. Linux/Unix系统SSH远程按Backspace键删除时出现^H的处理方法

    在linux/unix系统中连接SSH远程工作时,输出字符后按Backspace键删除时,会出现^H,这对习惯了按Backspace键删除的用户来说,感觉非常别扭,虽然可以通过Ctrl+Backspa ...

  2. [SinGuLaRiTy] Nescafe 24杯模拟赛

    [SinGularLaRiTy-1044] Copyright (c) SinGuLaRiTy 2017. All Rights Reserved. 小水塘(lagoon) 题目描述 忘川沧月的小水塘 ...

  3. Git学习(1)-本地版本库的创建

    我用的是Git-2.14.3-64-bit版本,在windows64位上运行的,把软件分享下链接:http://pan.baidu.com/s/1jIoZ7Xc 密码:13q2. 安装及配置自行百度, ...

  4. 再叙Java反射

    Java中的反射 本文为反射的基础知识部分. 能够分析类能力的程序被称为反射(reflective). 反射机制允许程序在运行时取得任何一个已知名称的class的内部信息,容许程序在运行时加载.探知. ...

  5. mysql查询语句处理

    两表做链接查询,   查理处理顺序各个阶段: 1) From: 对From子句中的坐标<left_table>和右表<right_table>执行笛卡尔积,产生虚拟表T1: 2 ...

  6. 2017-07-12(touch df du)

    touch 作用:修改文件时间或者创建新文件 文件三个主要的变动时间 mtime 当文件的内容发生改变的时候,就会更新这个时间 ctime 当文件的权限与属性发生改变的时候,就会更新这个时间 atim ...

  7. 使用SQL 提示优化sql

    use index 在查询语句中表名的后面,添加use index来提供希望mysql去参考的索引列表,就可以让mysql不再考虑其他可用的索引 explain select * from renta ...

  8. git fetch, git pull 剖析

    真正理解 git fetch, git pull 要讲清楚git fetch,git pull,必须要附加讲清楚git remote,git merge .远程repo, branch . commi ...

  9. junit4X系列--Rule

    原文出处:http://www.blogjava.net/DLevin/archive/2012/05/12/377955.html.感谢作者的无私分享. 初次用文字的方式记录读源码的过程,不知道怎么 ...

  10. Hyperledger Fabric Chaincode for Operators——实操智能合约

    什么是Chaincode(智能合约)? chaincode是一个程序,它是使用Go语言编写的,最终在Java等其他编程语言中实现了指定的接口.chaincode运行在一个被背书peer进程独立出来的安 ...