【BZOJ2595】游览计划(状压DP,斯坦纳树)
题意:见题面(我发现自己真是越来越懒了)
有N*M的矩阵,每个格子有一个值a[i,j]
现要求将其中的K个点(称为关键点)用格子连接起来,取(i,j)的费用就是a[i,j]
求K点全部连通的最小花费以及方案
n,m,k<=10
思路:斯坦纳树
虽然去年就疑似过了一道裸题,不过估计也是COPY的std,早就忘干净了
先%了一发论文,看到了几道有意思的SPFA的应用,准备去做一下
dp[i,j,sta]表示以(i,j)为根,关键点联通情况为sta的最小花费
显然初始化 \[ dp[i,j,1<<(k-1)]=0 (i,j为k号关键点)\]
\[ dp[i,j,sta]=dp[i,j,x]+dp[i,j,sta xor x]-a[i,j] \] 即合并子集
\[ dp[i,j,sta]=dp[x,y,sta]+a[i,j]\] 即合并道路
第一个转移可以枚举子集
第二个转移可能有环且形式是最短路,使用SPFA队列更新
类似一个分层图,SPFA的时候只用跑本层的内容
以下转自某大神blog:进行spfa的时候只需要对当前层的节点进行spfa就行了,不需要整个图完全松弛一遍,因为更高的层都可以通过枚举子集而变成若干个更低的层
时间复杂度:SPFA显然O(2^k)
枚举子集的dp时每个点有3个状态:不是子集,是子集但没取到,是子集且枚举到了
所以O(3^k)
总时间复杂度O(3^k*n*m)
PS:其实暴力的想法:Sigma C(k,k-i)*2^i用二项式展开就是3^k啦(感谢邻桌数学国家队CWY同学)
const oo=;
dx:array[..]of longint=(-,,,);
dy:array[..]of longint=(,,-,);
var dp:array[..,..,..]of longint;
pre:array[..,..,..,..]of longint;
flag,a,inq:array[..,..]of longint;
q:array[..,..]of longint;
n,m,i,j,x,y,k,v,k1,t,w,sta,tmp,ux,uy:longint; procedure dfs(i,j,sta:longint);
var x,y,z:longint;
begin
if sta= then exit;
flag[i,j]:=;
x:=pre[i,j,sta,]; y:=pre[i,j,sta,]; z:=pre[i,j,sta,];
dfs(x,y,z);
if (x=i)and(y=j) then dfs(x,y,sta xor z);
end; procedure print(x,y:longint);
var i,j:longint;
begin
writeln(dp[x,y,(<<k1)-]);
fillchar(flag,sizeof(flag),);
dfs(x,y,(<<k1)-);
for i:= to n do
begin
for j:= to m do
if a[i,j]> then
begin
if flag[i,j]= then write('o')
else write('_');
end
else write('x');
writeln;
end;
end; begin
assign(input,'bzoj2595.in'); reset(input);
assign(output,'bzoj2595.out'); rewrite(output);
readln(n,m);
for i:= to n do
for j:= to m do read(a[i,j]);
fillchar(dp,sizeof(dp),$7f);
for i:= to n do
for j:= to m do
if a[i,j]= then
begin
inc(k1); dp[i,j,<<(k1-)]:=;
end;
for v:= to (<<k1)- do
begin
fillchar(inq,sizeof(inq),);
t:=; w:=;
for i:= to n do
for j:= to m do
begin
x:=v and (v-);
while x> do
begin
tmp:=dp[i,j,x]+dp[i,j,v xor x]-a[i,j];
if tmp<dp[i,j,v] then
begin
dp[i,j,v]:=tmp;
pre[i,j,v,]:=i; pre[i,j,v,]:=j; pre[i,j,v,]:=x;
end;
x:=v and (x-);
end;
if dp[i,j,v]<oo then begin inc(w); q[w,]:=i; q[w,]:=j; inq[i,j]:=; end;
end;
while t<w do
begin
inc(t); ux:=q[t mod ,]; uy:=q[t mod ,]; inq[ux,uy]:=;
for k:= to do
begin
x:=ux+dx[k]; y:=uy+dy[k];
if (x>)and(x<=n)and(y>)and(y<=m)and(dp[ux,uy,v]+a[x,y]<dp[x,y,v]) then
begin
dp[x,y,v]:=dp[ux,uy,v]+a[x,y];
pre[x,y,v,]:=ux; pre[x,y,v,]:=uy; pre[x,y,v,]:=v;
if inq[x,y]= then
begin
inc(w); q[w mod ,]:=x; q[w mod ,]:=y; inq[x,y]:=;
end;
end;
end;
end;
end; for i:= to n do
begin
for j:= to m do
if a[i,j]= then begin print(i,j); break; end;
if a[i,j]= then break;
end;
close(input);
close(output);
end.
【BZOJ2595】游览计划(状压DP,斯坦纳树)的更多相关文章
- luogu4294 [WC2008]游览计划(状压DP/斯坦纳树)
link 题目大意:给定一个网格图,有些点是关键点,选择格点有代价,求把所有关键点联通的最小代价 斯坦纳树模板题 斯坦纳树问题:给定一个图结构,有一些点是关键点,求把这些关键点联通的最小代价e 斯坦纳 ...
- [BZOJ4006][JLOI2015]管道连接 状压dp+斯坦纳树
4006: [JLOI2015]管道连接 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 1020 Solved: 552[Submit][Statu ...
- [WC2008]游览计划 状压DP,斯坦纳树
---题面--- 题解: 这是一道斯坦纳树的题,用状压+spfa来解决 什么是斯坦纳树? 一开始还以为是数据结构来着,其实跟最小生成树很像,大致就是最小生成树只能在各个点之间直接相连,而斯坦纳树则允许 ...
- [WC2008]游览计划(状压dp)
题面太鬼畜不粘了. 题意就是给一张n*m的网格图,每个点有点权,有k个关键点,让你把这k个关键点连成一个联通快的最小代价. 题解 这题nmk都非常小,解法肯定是状压,比较一般的解法插头dp,但不太好写 ...
- [bzoj2595][WC2008]游览计划/[bzoj5180][Baltic2016]Cities_斯坦纳树
游览计划 bzoj-2595 wc-2008 题目大意:题目链接.题目连接. 注释:略. 想法:裸题求斯坦纳树. 斯坦纳树有两种转移方式,设$f[s][i]$表示联通状态为$s$,以$i$为根的最小代 ...
- 动态规划:状压DP-斯坦纳树
最小生成树是最小斯坦纳树的一种特殊情况 最小生成树是在给定的点集和边中寻求最短网络使所有点连通 而最小斯坦纳树允许在给定点外增加额外的点,使生成的最短网络开销最小 BZOJ2595 题意是给定一个棋盘 ...
- BZOJ.2595.[WC2008]游览计划(DP 斯坦纳树)
题目链接 f[i][s]表示以i为根节点,当前关键点的连通状态为s(每个点是否已与i连通)时的最优解.i是枚举得到的根节点,有了根节点就容易DP了. 那么i为根节点时,其状态s的更新为 \(f[i][ ...
- 【状压dp】Trie 树 @中山纪念中学20170304
目录 Trie 树 PROBLEM 题目描述 输入 输出 样例输入 样例输出 SOLUTION CODE Trie 树 PROBLEM 题目描述 字母(Trie)树是一个表示一个字符串集合中所有字符串 ...
- HDU.3311.Dig The Wells(DP 斯坦纳树)
题目链接 \(Description\) 有n座庙.一共n+m个点,可以在任意一些点修建水井,不同位置花费不同:也可以某些点之间连无向边共享水.求使n座庙都有水的最小花费. \(Solution\) ...
- BZOJ2595 Wc2008 游览计划 【斯坦纳树】【状压DP】*
BZOJ2595 Wc2008 游览计划 Description Input 第一行有两个整数,N和 M,描述方块的数目. 接下来 N行, 每行有 M 个非负整数, 如果该整数为 0, 则该方块为一个 ...
随机推荐
- winsock教程- windows下的socket编程(c语言实现)
winsock教程- windows下的socket编程(c语言实现) 使用winsock进行socket 编程 这是一个学习windows下socket编程(c语言)的快速指南.这是因为一下 ...
- C#常见控件命名规则举例
控件 缩写 举例 Adrotator adrt adrtTopAd BulletedList blst blstCity Button btn btnSubmit Calendar ca ...
- WorldChat.lua --世界聊天
print(">>Script: Wolrd Chating.") local SAY=" "--普通聊天 --联盟阵营 --部落阵营 local ...
- 论文阅读之:Photo-Realistic Single Image Super-Resolution Using a Generative Adversarial Network
Photo-Realistic Single Image Super-Resolution Using a Generative Adversarial Network 2016.10.23 摘要: ...
- POJ1229 域名匹配
给你两个域名,域名中包含一些通配符. * :匹配一个或任意多个部分 ?:匹配一个或三个部分 !:匹配三个以上部分. 求这两个域名是否能够表示同一个域名? 域名的长度不超过255. 分析:设给出的域名为 ...
- RMAN_学习笔记5_RMAN Catalog Script恢复目录脚本
2014-12-24 Created By BaoXinjian
- Java锁 到底锁的是哪个对象?
更新:在一次和一位专家的交谈中,他对一下代码能否能够成功同步,给予了否定的答案, 他的理由是”以构造函数的成员变量作为synchronized的锁,在多线程的情况下,每一个线程都持有自己私有变量的锁, ...
- REST风格URL
以前就是觉得 /nowamagic/article/article_id 这样的地址非常的漂亮,但是那只是表象罢了,了解深入以后,发现必须有一个客户端的Ajax Engine和Server端的服务配合 ...
- Codeforces 724C [坐标][乱搞][模拟]
/* 不要低头,不要放弃,不要气馁,不要慌张 题意: 从(0,0)出发与x轴正方向呈45度角的射线,在给定的矩形区域内不断发射,直到射入矩形的某个角停止. 给出多个坐标,问光线最早经过某坐标的时间. ...
- python_day2
一.字符串的基本使用 #!/usr/bin/env python #!-*- coding:utf-8 -*- #!/usr/bin/env python 指定解释器为python abc='hel ...