题意:见题面(我发现自己真是越来越懒了)

有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,斯坦纳树)的更多相关文章

  1. luogu4294 [WC2008]游览计划(状压DP/斯坦纳树)

    link 题目大意:给定一个网格图,有些点是关键点,选择格点有代价,求把所有关键点联通的最小代价 斯坦纳树模板题 斯坦纳树问题:给定一个图结构,有一些点是关键点,求把这些关键点联通的最小代价e 斯坦纳 ...

  2. [BZOJ4006][JLOI2015]管道连接 状压dp+斯坦纳树

    4006: [JLOI2015]管道连接 Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 1020  Solved: 552[Submit][Statu ...

  3. [WC2008]游览计划 状压DP,斯坦纳树

    ---题面--- 题解: 这是一道斯坦纳树的题,用状压+spfa来解决 什么是斯坦纳树? 一开始还以为是数据结构来着,其实跟最小生成树很像,大致就是最小生成树只能在各个点之间直接相连,而斯坦纳树则允许 ...

  4. [WC2008]游览计划(状压dp)

    题面太鬼畜不粘了. 题意就是给一张n*m的网格图,每个点有点权,有k个关键点,让你把这k个关键点连成一个联通快的最小代价. 题解 这题nmk都非常小,解法肯定是状压,比较一般的解法插头dp,但不太好写 ...

  5. [bzoj2595][WC2008]游览计划/[bzoj5180][Baltic2016]Cities_斯坦纳树

    游览计划 bzoj-2595 wc-2008 题目大意:题目链接.题目连接. 注释:略. 想法:裸题求斯坦纳树. 斯坦纳树有两种转移方式,设$f[s][i]$表示联通状态为$s$,以$i$为根的最小代 ...

  6. 动态规划:状压DP-斯坦纳树

    最小生成树是最小斯坦纳树的一种特殊情况 最小生成树是在给定的点集和边中寻求最短网络使所有点连通 而最小斯坦纳树允许在给定点外增加额外的点,使生成的最短网络开销最小 BZOJ2595 题意是给定一个棋盘 ...

  7. BZOJ.2595.[WC2008]游览计划(DP 斯坦纳树)

    题目链接 f[i][s]表示以i为根节点,当前关键点的连通状态为s(每个点是否已与i连通)时的最优解.i是枚举得到的根节点,有了根节点就容易DP了. 那么i为根节点时,其状态s的更新为 \(f[i][ ...

  8. 【状压dp】Trie 树 @中山纪念中学20170304

    目录 Trie 树 PROBLEM 题目描述 输入 输出 样例输入 样例输出 SOLUTION CODE Trie 树 PROBLEM 题目描述 字母(Trie)树是一个表示一个字符串集合中所有字符串 ...

  9. HDU.3311.Dig The Wells(DP 斯坦纳树)

    题目链接 \(Description\) 有n座庙.一共n+m个点,可以在任意一些点修建水井,不同位置花费不同:也可以某些点之间连无向边共享水.求使n座庙都有水的最小花费. \(Solution\) ...

  10. BZOJ2595 Wc2008 游览计划 【斯坦纳树】【状压DP】*

    BZOJ2595 Wc2008 游览计划 Description Input 第一行有两个整数,N和 M,描述方块的数目. 接下来 N行, 每行有 M 个非负整数, 如果该整数为 0, 则该方块为一个 ...

随机推荐

  1. winsock教程- windows下的socket编程(c语言实现)

    winsock教程- windows下的socket编程(c语言实现) 使用winsock进行socket 编程     这是一个学习windows下socket编程(c语言)的快速指南.这是因为一下 ...

  2. C#常见控件命名规则举例

    控件       缩写  举例 Adrotator adrt adrtTopAd BulletedList blst blstCity Button btn btnSubmit Calendar ca ...

  3. WorldChat.lua --世界聊天

    print(">>Script: Wolrd Chating.") local SAY=" "--普通聊天 --联盟阵营 --部落阵营 local ...

  4. 论文阅读之: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 摘要: ...

  5. POJ1229 域名匹配

    给你两个域名,域名中包含一些通配符. * :匹配一个或任意多个部分 ?:匹配一个或三个部分 !:匹配三个以上部分. 求这两个域名是否能够表示同一个域名? 域名的长度不超过255. 分析:设给出的域名为 ...

  6. RMAN_学习笔记5_RMAN Catalog Script恢复目录脚本

    2014-12-24 Created By BaoXinjian

  7. Java锁 到底锁的是哪个对象?

    更新:在一次和一位专家的交谈中,他对一下代码能否能够成功同步,给予了否定的答案, 他的理由是”以构造函数的成员变量作为synchronized的锁,在多线程的情况下,每一个线程都持有自己私有变量的锁, ...

  8. REST风格URL

    以前就是觉得 /nowamagic/article/article_id 这样的地址非常的漂亮,但是那只是表象罢了,了解深入以后,发现必须有一个客户端的Ajax Engine和Server端的服务配合 ...

  9. Codeforces 724C [坐标][乱搞][模拟]

    /* 不要低头,不要放弃,不要气馁,不要慌张 题意: 从(0,0)出发与x轴正方向呈45度角的射线,在给定的矩形区域内不断发射,直到射入矩形的某个角停止. 给出多个坐标,问光线最早经过某坐标的时间. ...

  10. python_day2

    一.字符串的基本使用 #!/usr/bin/env python #!-*- coding:utf-8 -*- #!/usr/bin/env python  指定解释器为python abc='hel ...