@bzoj - 2595@ 游览计划
@description@
从未来过绍兴的小D有幸参加了Winter Camp 2008,他被这座历史名城的秀从未来过绍兴的小D有幸参加了Winter Camp 2008,他被这座历史名城的秀丽风景所吸引,强烈要求游览绍兴及其周边的所有景点。
主办者将绍兴划分为N行M列(NXM)个方块,景点含于方块内,且一个方块至多有一个景点。无景点的方块视为路。
为了保证安全与便利,主办方依据路况和治安状况,在非景点的一些方块内安排不同数量的志愿者:在景点内聘请导游(导游不是志愿者)。在选择旅游方案时,保证任意两个景点之间,存在一条路径,在这条路径所经过的每一个方块都有志愿者或者该方块为景点。既能满足选手们游览的需要,又能够让志愿者的总数最少。
现在,希望你能够帮助主办方找到一种最好的安排方案。
@solution@
斯坦纳树经典题。
斯坦纳树的定义是:给定一个图与 k 个特殊点,求将这 k 个特殊点连通的最小生成树。
当 k = 点数时,就是常见的最小生成树的定义。
状压 dp:记 dp[i][s] 表示当前点为 i,已经包含了 k 个特殊点的集合为 s,最小连通代价。
第一类转移:dp[i][s] = min{dp[i][t] + dp[i][s xor t]},即两个集合取并集。
第二类转移:dp[i][s] = min{dp[j][s] + dis(i, j)},即任意加一条边。
正确性很显然,因为每一种可行方案都会被上面的过程考虑到。
第二类转移带环,但是注意到这个转移很像最短路的 “松弛” 操作,于是我们直接写个 spfa 即可。
时间复杂度(如果把 spfa 算作 O(nm) 的上界的话)为 O(n*3^k + nm*2^k)。
本题因为是点权,为了不使第一种转移出现重复,dp[i][s] 不包含 i 的点权。
@accepted code@
#include <queue>
#include <cstdio>
#include <iostream>
#include <algorithm>
using namespace std;
typedef pair<int, int> pii;
#define mp make_pair
#define fi first
#define se second
const int INF = (1 << 30);
const int dx[] = {0, 1, -1, 0, 0};
const int dy[] = {0, 0, 0, -1, 1};
bool inq[10][10];
int A[10][10], id[10][10], N, M, K, S;
int dp[10][10][1 << 10], pre[10][10][1 << 10];
queue<pii>que;
void run(int s) {
for(int i=0;i<N;i++)
for(int j=0;j<M;j++)
que.push(mp(i, j)), inq[i][j] = true;
while( !que.empty() ) {
pii p = que.front(); que.pop(); inq[p.fi][p.se] = false;
for(int i=1;i<=4;i++) {
int x0 = p.fi + dx[i], y0 = p.se + dy[i];
if( x0 >= N || y0 >= M || x0 < 0 || y0 < 0 ) continue;
if( dp[x0][y0][s] > dp[p.fi][p.se][s] + A[p.fi][p.se] ) {
dp[x0][y0][s] = dp[p.fi][p.se][s] + A[p.fi][p.se];
pre[x0][y0][s] = i;
if( !inq[x0][y0] ) {
que.push(mp(x0, y0));
inq[x0][y0] = true;
}
}
}
}
}
void solve() {
for(int s=1;s<S;s++) {
for(int i=0;i<N;i++)
for(int j=0;j<M;j++) {
int t = s & (s - 1);
while( t ) {
if( dp[i][j][s] > dp[i][j][t] + dp[i][j][s ^ t] ) {
dp[i][j][s] = dp[i][j][t] + dp[i][j][s ^ t];
pre[i][j][s] = -t;
}
t = s & (t - 1);
}
}
run(s);
}
}
bool tag[10][10];
void get(int x, int y, int s) {
tag[x][y] = true;
if( pre[x][y][s] > 0 )
get(x - dx[pre[x][y][s]], y - dy[pre[x][y][s]], s);
else if( pre[x][y][s] < 0 ) {
pre[x][y][s] = -pre[x][y][s];
get(x, y, pre[x][y][s]), get(x, y, s^pre[x][y][s]);
}
}
int main() {
scanf("%d%d", &N, &M);
for(int i=0;i<N;i++)
for(int j=0;j<M;j++) {
scanf("%d", &A[i][j]);
if( A[i][j] == 0 ) id[i][j] = (K++);
}
S = (1 << K);
for(int i=0;i<N;i++)
for(int j=0;j<M;j++) {
for(int s=0;s<S;s++) dp[i][j][s] = INF;
if( A[i][j] == 0 ) dp[i][j][1 << id[i][j]] = 0;
}
solve();
int x, y, ans = INF;
for(int i=0;i<N;i++)
for(int j=0;j<M;j++)
if( ans > dp[i][j][S - 1] + A[i][j] )
ans = dp[i][j][S - 1] + A[i][j], x = i, y = j;
printf("%d\n", ans);
get(x, y, S - 1);
for(int i=0;i<N;i++) {
for(int j=0;j<M;j++) {
if( tag[i][j] ) {
if( A[i][j] == 0 )
putchar('x');
else putchar('o');
}
else putchar('_');
}
puts("");
}
}
/*
8 8
1 4 1 3 4 2 4 1
4 3 1 2 0 1 2 3
3 2 1 3 0 3 1 2
2 6 5 0 2 4 1 0
5 1 2 1 3 4 2 5
5 1 3 1 5 0 1 4
5 0 6 1 4 5 3 4
0 2 2 2 3 4 1 1
*/
@details@
关于第二类转移,其实也可以用 floyd 先预处理出所有点对的最短路,然后直接点对两两 O(n^2) 转移。
因为 spfa 的复杂度上界是 O(NM),对于 M 比较大的稠密图用 floyd 更快。
当然对于本题这种网格图 spfa 的复杂度和 floyd 没差,spfa 的常数说不定还要小一些。
@bzoj - 2595@ 游览计划的更多相关文章
- 【BZOJ 2595】2595: [Wc2008]游览计划 (状压DP+spfa,斯坦纳树?)
2595: [Wc2008]游览计划 Time Limit: 10 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 1572 Solved: 7 ...
- 【BZOJ】【2595】【WC2008】游览计划
Orz zky神犇http://blog.csdn.net/iamzky/article/details/42029921 spfa的灵活应用!(好像是求了一个叫做斯坦纳树的东西……) o(︶︿︶)o ...
- 【BZOJ-2595】游览计划 斯坦纳树
2595: [Wc2008]游览计划 Time Limit: 10 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 1518 Solved: 7 ...
- 【LG4294】[WC2008]游览计划
[LG4294][WC2008]游览计划 题面 洛谷 bzoj 题解 斯坦纳树板子题. 斯坦纳树的总结先留个坑. 代码 #include <iostream> #include <c ...
- [bzoj2595][WC2008]游览计划/[bzoj5180][Baltic2016]Cities_斯坦纳树
游览计划 bzoj-2595 wc-2008 题目大意:题目链接.题目连接. 注释:略. 想法:裸题求斯坦纳树. 斯坦纳树有两种转移方式,设$f[s][i]$表示联通状态为$s$,以$i$为根的最小代 ...
- BZOJ_2595_[Wc2008]游览计划_斯坦纳树
BZOJ_2595_[Wc2008]游览计划_斯坦纳树 题意: 分析: 斯坦纳树裸题,有几个需要注意的地方 给出矩阵,不用自己建图,但枚举子集转移时会算两遍,需要减去当前点的权值 方案记录比较麻烦,两 ...
- [WC2008]游览计划 解题报告
[WC2008]游览计划 斯坦纳树板子题,其实就是状压dp 令\(dp_{i,s}\)表示任意点\(i\)联通关键点集合\(s\)的最小代价 然后有转移 \[ dp_{i,S}=\min_{T\in ...
- bzoj2595 / P4294 [WC2008]游览计划
P4294 [WC2008]游览计划 斯坦纳树 斯坦纳树,是一种神奇的树.它支持在一个连通图上求包含若干个选定点的最小生成树. 前置算法:spfa+状压dp+dfs(大雾) 我们设$f[o][P]$为 ...
- 【BZOJ2595】 [Wc2008]游览计划
BZOJ2595 [Wc2008]游览计划 Solution 考虑这是一个最小费用连通性的问题,既然大家都说这是什么斯坦纳树那就是的吧... 所以我们肯定可以这样设一个dp状态: \(dp_{i,j, ...
随机推荐
- 使用element-ui 的table 组件 出现表格线条不对齐的问题
在全局css样式中添加以下代码即可: body .el-table th.gutter { display: table-cell !important }
- 【HTTP】HTTP报文&状态码
HTTP报文中的HTTP信息 一.编码提升传输速率 编码的好处:有效处理大量的访问请求 编码的弊端:会消耗更多的CPU资源 报文主体&实体主体 报文:HTTP通信的基本单元,8位组字节流组成, ...
- [CSS布局基础]居中布局的实现方式总结
[原创]码路工人 Coder-Power 大家好,这里是码路工人有力量,我是码路工人,你们是力量. github-pages 博客园cnblogs 做Web开发少不了做页面布局.码路工人给大家总结一下 ...
- Linux学习(二):makefile
编译命令: gcc -o exefile src.c (将src.c编译,链接为exefile可执行文件) gcc -o obj.o -c src.c (将src.c编译为obj.o目标文件) mak ...
- 国家集训队 部落战争 网络流最小路径覆盖 洛谷P2172
洛谷AC传送门! step1: 题目大意 有一张M x N的网格图,有一些点为“ * ”可以走,有一些点为“ x ”不能走,每走一步你都可以移动R * C 个格子(参考象棋中马的走法),且不能回头,已 ...
- 一个 json 转换工具
在前后端的数据协议(主要指http和websocket)的问题上,如果前期沟通好了,那么数据协议上问题会很好解决,前后端商议一种都可以接受的格式即可.但是如果接入的是老系统.第三方系统,或者由于某些奇 ...
- day 6 (笔记由图转)
.
- Chisel3 - Tutorial - Tbl
https://mp.weixin.qq.com/s/e8vJ8claauBtiuedxYYaJw 实现可以动态索引的表. 参考链接: https://github.com/ucb-bar/c ...
- [C#.NET拾遗补漏]01:字符串操作
字符串操作在任意编程语言的日常编程中都随处可见,今天来汇总一下 C# 中关于字符串的一些你可能遗忘或遗漏的知识点. 逐字字符串 在普通字符串中,反斜杠字符是转义字符.而在逐字字符串(Verbatim ...
- Java实现 蓝桥杯VIP 基础练习 回形取数
问题描述 回形取数就是沿矩阵的边取数,若当前方向上无数可取或已经取过,则左转90度.一开始位于矩阵左上角,方向向下. 输入格式 输入第一行是两个不超过200的正整数m, n,表示矩阵的行和列.接下来m ...