BZOJ 2595: [Wc2008]游览计划 [DP 状压 斯坦纳树 spfa]【学习笔记】
题意:略
http://www.cnblogs.com/lazycal/p/bzoj-2595.html
本题的核心就是求斯坦纳树:
Steiner Tree:
Given an undirected graph with non-negative edge weights and a subset of vertices, usually referred to as terminals,
the Steiner tree problem in graphs requires a tree of minimum weight that contains all terminals (but may include additional vertices).
也就是对于给定的点集求一颗包含他的最小生成树(可以包含额外的点)
$ST$是$NPC$问题,规模小的情况可以使用状压$DP$解决
$f[i][s]$表示根在$i$,连通的点集为$s$的(仅包括给定点集中的点)的最小花费
有两种转移:
对于点权的情况(边权类似):
$f[i][s]=min{f[i][s']+f[i][s-s']-val[i]}$,划分成两个子集,具有阶段性普通$DP$就可以
$f[i][s]=min{f[i'][s]+val[i]}$,从一颗树扩展而来,阶段性不明显,但满足三角不等式,使用$spfa$求解
那么过程就很清楚了
- 从小到大枚举集合和点
- 第一种转移枚举子集
- 第二种转移对当前集合使用spfa
然后就到黄学长哪里仿写了份模板
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <queue>
using namespace std;
#define pii pair<int,int>
#define MP make_pair
#define fir first
#define sec second
typedef long long ll;
const int N=,S=(<<)+,INF=1e9;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
} int n,m,k,a[N][N];
int f[N][N][S];
struct Path{
int i,j,s;
Path(){}
Path(int a,int b,int c):i(a),j(b),s(c){}
}pre[N][N][S]; queue<pii> q;
bool inq[N][N];
int dx[]={,-,,},dy[]={,,,-};
void spfa(int s){
while(!q.empty()){
int x=q.front().fir,y=q.front().sec;
inq[x][y]=;q.pop();
for(int k=;k<;k++){
int i=x+dx[k],j=y+dy[k];
if(i<||i>n||j<||j>m) continue;
if(f[i][j][s]>f[x][y][s]+a[i][j]){
f[i][j][s]=f[x][y][s]+a[i][j];
pre[i][j][s]=Path(x,y,s);
if(!inq[i][j])
q.push(MP(i,j)),inq[i][j]=;
}
}
}
}
bool vis[N][N];
void dfs(int x,int y,int s){
vis[x][y]=;
Path t=pre[x][y][s];
if(t.i==&&t.j==) return;
dfs(t.i , t.j , t.s);
if(t.i==x && t.j==y) dfs(t.i , t.j , s-t.s);
}
int main(){
freopen("in","r",stdin);
n=read();m=read();
memset(f,0x3f,sizeof(f));
for(int i=;i<=n;i++)
for(int j=;j<=m;j++){
a[i][j]=read();
if(!a[i][j]) f[i][j][<<k]=,k++;
} int All=<<k;
for(int sa=;sa<All;sa++){
for(int i=;i<=n;i++)
for(int j=;j<=m;j++){
for(int s=sa&(sa-);s;s=sa&(s-)){
int _=f[i][j][s]+f[i][j][sa-s]-a[i][j];
if(_<f[i][j][sa]){
f[i][j][sa]=_;
pre[i][j][sa]=Path(i,j,s);
}
}
if(f[i][j][sa]<INF) q.push(MP(i,j)),inq[i][j]=;
}
spfa(sa);
} int x=,y=,flag=;
for(int i=;i<=n&&!flag;i++)
for(int j=;j<=m;j++) if(!a[i][j]) {x=i;y=j;flag=;break;}
dfs(x,y,All-);
printf("%d\n",f[x][y][All-]);
for(int i=;i<=n;i++){
for(int j=;j<=m;j++){
if(a[i][j]==) putchar('x');
else if(vis[i][j]) putchar('o');
else putchar('_');
}
puts("");
}
}
BZOJ 2595: [Wc2008]游览计划 [DP 状压 斯坦纳树 spfa]【学习笔记】的更多相关文章
- 【BZOJ 2595】2595: [Wc2008]游览计划 (状压DP+spfa,斯坦纳树?)
2595: [Wc2008]游览计划 Time Limit: 10 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 1572 Solved: 7 ...
- [bzoj2595][WC2008]游览计划/[bzoj5180][Baltic2016]Cities_斯坦纳树
游览计划 bzoj-2595 wc-2008 题目大意:题目链接.题目连接. 注释:略. 想法:裸题求斯坦纳树. 斯坦纳树有两种转移方式,设$f[s][i]$表示联通状态为$s$,以$i$为根的最小代 ...
- BZOJ.2595.[WC2008]游览计划(DP 斯坦纳树)
题目链接 f[i][s]表示以i为根节点,当前关键点的连通状态为s(每个点是否已与i连通)时的最优解.i是枚举得到的根节点,有了根节点就容易DP了. 那么i为根节点时,其状态s的更新为 \(f[i][ ...
- bzoj:2595: [Wc2008]游览计划
Description Input 第一行有两个整数,N和 M,描述方块的数目. 接下来 N行, 每行有 M 个非负整数, 如果该整数为 0, 则该方块为一个景点:否则表示控制该方块至少需要的志愿者数 ...
- bzoj 2595 [Wc2008]游览计划(斯坦纳树)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2595 [题意] 给定N*M的长方形,选最少权值和的格子使得要求的K个点连通. [科普] ...
- BZOJ 2595 [Wc2008]游览计划 ——斯坦纳树
[题目分析] 斯坦纳树=子集DP+SPFA? 用来学习斯坦纳树的模板. 大概就是用二进制来表示树包含的点,然后用跟几点表示树的形态. 更新分为两种,一种是合并两个子集,一种是换根,换根用SPFA迭代即 ...
- BZOJ2595 [Wc2008]游览计划 【状压dp + 最短路】
题目链接 BZOJ2595 题解 著名的斯坦纳树问题 设\(f[i][j][s]\)表示点\((i,j)\)与景点联通状况为\(s\)的最小志愿者数 设\(val[i][j]\)为\((i,j)\)需 ...
- 【BZOJ】2595: [Wc2008]游览计划
题意 \(n * m\)的网格,如果\(a_{i, j} = 0\)则表示景点,否则表示这里的需要的志愿者人数.求一种安排志愿者的方案使得所有景点连通且志愿者最少. 分析 本题可以插头dp,然而有一个 ...
- 【BZOJ2595】游览计划(状压DP,斯坦纳树)
题意:见题面(我发现自己真是越来越懒了) 有N*M的矩阵,每个格子有一个值a[i,j] 现要求将其中的K个点(称为关键点)用格子连接起来,取(i,j)的费用就是a[i,j] 求K点全部连通的最小花费以 ...
随机推荐
- [学习OpenCV攻略][006][平滑图片]
cvCreateImage(图片大小,像素位数,通道数) 创建图片,根据输入的图片大小,各个通道像素点的位数,和通道数.像素点宏IPL_DEPTH_8U cvGetSize(图片) 得到图片的大小信息 ...
- VisualSVN Server启动错误(0x8007042a)
SVN Server启动错误(0x8007042a) 原因是SVN Server端口被占用 打开VisualSVN Server, 菜单->操作->Properties->Net ...
- ssh密码
- PHP defined() 函数
定义和用法 defined() 函数检查某常量是否存在. 若常量存在,则返回 true,否则返回 false. 语法 defined(name) 参数 描述 name 必需.规定要检查的常量的名称. ...
- DEDECMS去掉自动生成首页或栏目后面带的index.html
Dede默认生成首页后,首页的链接后面会多出一个index.html.据官方说法这样有利于网站优化.但是这个index.html怎么看都不舒服,而且也不利于seo中主页url的统一.因为我的网站的ur ...
- linux nvme的那些workqueue
目前nvme三个常见的使用的workqueue ,主要有nvme_workq,nvme_rdma_wq ,nvme_fc_wq,下面一一描述一下初始化及使用的场景.分别对应于NVME over PCI ...
- 2017-07-12(touch df du)
touch 作用:修改文件时间或者创建新文件 文件三个主要的变动时间 mtime 当文件的内容发生改变的时候,就会更新这个时间 ctime 当文件的权限与属性发生改变的时候,就会更新这个时间 atim ...
- python_11_字符编码
什么是字符编码? --世界上有很多国家,每个国家都有自己独特的语言,所以在计算机普及的当今世界, 每个国家都有自己的字符编码,本国的软件运行在其他国家的机器上,会出现乱码, 有utf-8,gbk等各种 ...
- Windows核心编程&内存管理
1. 每个进程都有自己的虚拟地址空间,对于32位机器而言,这个地址空间的大小为4GB(2^32 / 1024^3),这个虚拟地址空间只不过是一个内存地址空间, 为了能够正常读/写数据,我们还需要把物理 ...
- [Qt Quick] No rule to make target问题解决办法
[问题描述] 修改项目中资源的qml文件名或删除无用资源文件后,重新构建项目时,会出现类似如下的问题提示: No rule to make target 'aaa', needed by 'bbb'. ...