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

有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. Mac OS X 上Lua的安装方法

    先在Mac OS的终端查询下本机是否已安装Lua Last login: Thu Jul 10 07:54:48 on ttys000 keshans-Mac-mini:~ keshan$ lua - ...

  2. Learning to write a compiler

    http://stackoverflow.com/questions/1669/learning-to-write-a-compiler?rq=1 Big List of Resources: A N ...

  3. phpunit4.1的干净测试

    一般而言,写测试时需要加载一些文件来进行自动加载 但在phpunit4.1中只要其中一个测试文件加载了,其他测试文件就不需要再加载

  4. Unity3D研究院之获取摄像机的视口区域

    摄像机分为两种,一种是正交摄像机还有一种是透视摄像机.正交摄像机无论远近它的视口范围永远是固定的,但是透视摄像机是由原点向外扩散性发射,也就是距离越远它的视口区域也就越大.那么我们如何获取距离摄像机任 ...

  5. PHP和Golang使用Thrift1和Thrift2访问Hbase0.96.2(ubuntu12.04)

    目录: 一.Thrift1和Thrift2的简要介绍 1) 写在前面 2) Thrift1和Thrift2的区别  二.Thrift0.9.2的安装 1) 安装依赖插件 2) Thrift0.9.2的 ...

  6. 抽象类和接口 static final finalize

    抽象类 任何含有一个或多个抽象方法的类都必须声明成抽象类,用关键字abstract. 抽象类没有对象,即一个抽象类不能用new来直接实例化. 不能定义抽象构造方法或抽象静态方法.(但是可以创建父类引用 ...

  7. iOS学习-----真机测试过程

    原文 http://www.th7.cn/Program/IOS/201407/233103.shtml   1:进入苹果开发者平台 2:进入Member Center 3:输入开发者账号和密码 4: ...

  8. 《MySQL 存储过程编程》-读书笔记

    本书结构: 第一部分:存储编程基础 第1章:存储过程程序基础 第2章:MySQL存储编程指南 第3章:语言基础 第4章:语句块 第5章:在存储程序中使用SQL 第一章:MySQL存储程序介绍 存储程序 ...

  9. Eclipse下新建Maven项目、自动打依赖jar包

    当我们无法从本地仓库找到需要的构件的时候,就会从远程仓库下载构件至本地仓库.一般地,对于每个人来说,书房只有一个,但外面的书店有很多,类似第,对于Maven来说,每个用户只有一个本地仓库,但可以配置访 ...

  10. 也谈Oracle异常: ORA-06502: PL/SQL: 数字或值错误 : 字符串缓冲区太小

    场景: .Net程序调用Oracle方法 command.ExecuteNonQuery()执行存储过程;弹出如标题异常信息. 存储过程有4个输入参数,一个输出参数.一旦执行到给输出参数赋值的时候就报 ...