洛谷 P1171 售货员的难题
题目背景
数据有更改
题目描述
某乡有n个村庄(1<n<20),有一个售货员,他要到各个村庄去售货,各村庄之间的路程s(0<s<1000)是已知的,且A村到B村与B村到A村的路大多不同。为了提高效率,他从商店出发到每个村庄一次,然后返回商店所在的村,假设商店所在的村庄为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到各村的路程}
80分的暴力:
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
int n,tot;
int vis[];
int map[][];
int minn=0x7f7f7f7f,ans=0x7f7f7f7f;
void dfs(int now,int num,int dis){
if(num==n){
ans=min(ans,dis+map[now][]);
return ;
}
if(dis+n-num-+minn>=ans) return ;
for(int i=;i<=n;i++)
if(!vis[i]){
vis[i]=;
dfs(i,num+,dis+map[now][i]);
vis[i]=;
}
}
int main(){
scanf("%d",&n);
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
scanf("%d",&map[i][j]);
for(int i=;i<=n;i++)
minn=min(minn,map[i][]);
vis[]=;
dfs(,,);
cout<<ans;
}
/*
3
0 1 2
2 0 1
1 2 0
*/
正解:
状压的思路是一样的。用 f [ i ] [ j ] 来表示 i 状态下走到第 j 个地方的最小值。这里的 i 实质上是一个二进制数,每一位是 0 是 1 即表示每个地方有无去过,但是转为十进制表示状态,这便是状态压缩的基本思想。先从 3(二进制 11) 枚举 i,每次给 i 加 2(因为第一位所表示的第一个地方是起点,不管如何都去过,因此其永远是 1)。得到可能的 i 后,枚举 i 的除第一位外每个为 1 的位,并替换 1 为 0 得到能转移到状态 i 的状态 s,具体转移过程就不多说了,总之位运算什么的详见代码。
那么如何进行优化呢?下面就是几个好办法:
1 . 首先如果规定 n = 5,即有售货员要去五个地方,枚举到 i = 3(二进制 00011) 时,我们不一定需要从最低位一直枚举到第 n 位,因为第 n 位可能在枚举 i 的很久以后才能变成 1,这之前都是 0,浪费时间复杂度,因此我们可以规定整数 k,表示目前可能为 1 的最高位的位数。当 i 超过 2 的 k 次方时,更新 k,即为 k 自增。这里 2 的 k 次方可以暂时用变量 p 表示,k 更新时用位运算给 p 向左移一位。
2 . 尽量不用 STL 的 min,虽然好用,但是宁愿用 define 手打 QAQ,另外其他联系到位运算的,比如取某数二进制位下的某位的值,也可以用 define 而不是新建什么内联函数。
- 3 . 对于状态 i,其由不同的状态 s 转移而来,因此,我们倒推 s 的时候,先确认其可行性,再枚举 l ,用 f [ s ] [ l ] 更新 f [ i ] [ j ] 的最小值。
个人认为第 2 点优化程度是最大的。下面给出代码:
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define min(a,b) a<b?a:b
using namespace std;
int n,m,ans=0x7f7f7f7f;
int map[][],f[<<][];
int main(){
scanf("%d",&n);
m=(<<n)-;
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
scanf("%d",&map[i][j]);
memset(f,0x7f,sizeof(f));
f[][]=;
for(int i=,k=,p=;i<=m;i+=){
if(i>p) p=p<<,k++;
for(int j=;j<=k;j++)
if((i>>j-)&){
int s=i^(<<j-);
for(int l=;l<j;l++)
f[i][j]=min(f[i][j],f[s][l]+map[l][j]);
for(int l=j+;l<=k;l++)
f[i][j]=min(f[i][j],f[s][l]+map[l][j]);
}
}
for(int i=;i<=n;i++)
ans=min(ans,f[m][i]+map[i][]);
cout<<ans;
}
洛谷 P1171 售货员的难题的更多相关文章
- 洛谷P1171 售货员的难题
P1171 售货员的难题 题目背景 数据有更改 题目描述 某乡有n个村庄(1<n<20),有一个售货员,他要到各个村庄去售货,各村庄之间的路程s(0<s<1000)是已知的,且 ...
- 洛谷 P1171 售货员的难题 【状压dp】
题目描述 某乡有n个村庄(1<n<20),有一个售货员,他要到各个村庄去售货,各村庄之间的路程s(0<s<1000)是已知的,且A村到B村与B村到A村的路大多不同.为了提高效率 ...
- 洛谷P1171 售货员的难题【状压DP】
题目描述 某乡有n个村庄(1 输入格式: 村庄数n和各村之间的路程(均是整数). 输出格式: 最短的路程. 输入样例: 3 0 2 1 1 0 2 2 1 0 输出样例 3 说明 输入解释 3 {村庄 ...
- 2018.07.18 洛谷P1171 售货员的难题(状压dp)
传送门 感觉是一道经典的状压dp,随便写了一发卡了卡常数开了个O(2)" role="presentation" style="position: relati ...
- P1171 售货员的难题
P1171 售货员的难题 题目描述 某乡有nn个村庄(1<n \le 201<n≤20),有一个售货员,他要到各个村庄去售货,各村庄之间的路程s(0<s<1000)s(0< ...
- 洛谷 P1379 八数码难题 Label:判重&&bfs
特别声明:紫书上抄来的代码,详见P198 题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示.空格周围的棋子可以移到空格中.要求解的问题是:给 ...
- 【题解】P1171 售货员的难题
Tags 搜索,状压. 裸的旅行商问题 #include <stdio.h> #include <string.h> #define re register #define ...
- P1171 售货员的难题--搜索(剪枝)
题目背景 数据有更改 题目描述 某乡有nn个村庄(1<n \le 201<n≤20),有一个售货员,他要到各个村庄去售货,各村庄之间的路程s(0<s<1000)s(0<s ...
- 洛谷P1379八数码难题
题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示.空格周围的棋子可以移到空格中. 要求解的问题是:给出一种初始布局(初始状态)和目标布局(为 ...
随机推荐
- 【转载】HTTP Session 内存到内存复制的拓扑结构
http://www.oschina.net/question/129540_23215 HTTP 协议本身是“连接 - 请求 - 应答 - 关闭连接”的模式,是一种无状态协议:然而随着 web 动态 ...
- centos下nginx配置
转自 http://www.linuxidc.com/Linux/2016-09/134907.htm 安装所需环境 Nginx 是 C语言 开发,建议在 Linux 上运行,当然,也可以安装 Wi ...
- 使用dispatch_group来进行线程同步
我的上篇文章iOS中多个网络请求的同步问题总结中用到了dispatch_group来进行线程同步,对用法不是特别熟悉所以整理这篇文章来加深记忆(闲着也是闲着). 一.简单介绍下将会用到的一些东西 英语 ...
- Git常见问题 资料汇总
来源https://blog.csdn.net/albb_/article/details/80420468
- Windows server 2012R清除并重建SID 用于制作封装模板
首先介绍下什么是SID SID也就是安全标识符(Security Identifiers),是标识用户.组和计算机帐户的唯一的号码.在第一次创建该帐户时,将给网络上的每一个帐户发布一个唯一的 SID. ...
- Vue内置指令
v-text 类型: string用法: 更新元素的 textContent.如果要更新部分的 textContent ,需要使用 {{ Mustache }} 插值. v-html 类型: stri ...
- 【BZOJ 1089】[SCOI2003]严格n元树
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 设fi表示深度为i的树个数,si是fi的前缀和,即si为深度不超过i树的个数. 那么si=s[i-1]^n + 1 就是说 先选一个 ...
- EBS OAF开发中怎样通过ReferenceAO进行验证
EBS OAF开发中怎样通过ReferenceAO进行验证 (版权声明.本人原创或者翻译的文章如需转载,如转载用于个人学习,请注明出处:否则请与本人联系,违者必究) Reference AO 除了用于 ...
- Opencv(3.0.0beta)+Python(2.7.8 64bit) 简单具体,一遍成功
看到非常多配置的文章,都没法正常走完流程 使用到的资源,都是今天为止最新的: python-2.7.8.amd64.msi opencv-3.0.0-beta.exe numpy-MKL-1.9.1. ...
- 使用MyEclipse编写Java程序
MyEclipse是非常实用的一款Java程序开发工具,主要用于Java.Java EE以及移动应用的开发.MyEclipse的功能非常强大,支持也十分广泛,尤其是对各种开源产品的支持相当不错. My ...