斯坦纳树居然不是数据结构...

斯坦纳树应该是类似最小生成树的问题,只不过它允许增加额外的点

比如这道题就是找关于景点的最小生成树,但还要付出最小的代价,加一些额外的格子把景点都连起来

套路是设$f_{s,i}$表示至少选$s$这个集合的点,必经$i$的最小价值,那么转移就是$f_{s,i}\gets f_{u,i}+f_{s-u,i}-v_i(u\subset s)$和$f_{s,i}\gets f_{s,j}+w_{i,j}\left(\exists(i,j)\right)$

如果$f_{u,i}$和$f_{s-u,i}$选取的额外点有重合的部分,那么第一个转移算重了这些点的权值,所以还要加一个朴素的转移

第一个转移直接枚举子集即可,第二个转移用最短路来转移

这题应该就是斯坦纳树的裸题,要输出方案的话,记每个状态由什么转移过来,最后dfs一遍即可即可

#include<stdio.h>
#include<string.h>
typedef int mt[11][11];
const int go[4][2]={{0,1},{0,-1},{1,0},{-1,0}};
bool min(int&a,int b){
	if(b<a){
		a=b;
		return 1;
	}
	return 0;
}
mt a,f[1024];
int n,m,head,tail;
bool inq[11][11];
struct zt{
	int x,y;
	zt(int a=0,int b=0){x=a;y=b;}
}q[100010],t;
struct fr{
	int s,x,y;
	fr(int a=0,int b=0,int c=0){s=a;x=b;y=c;}
	bool hav(){return s|x|y;}
}g[1024][11][11];
void bfs(mt&dis){
	int i,x,y;
	while(head<=tail){
		t=q[head];
		head++;
		inq[t.x][t.y]=0;
		for(i=0;i<4;i++){
			x=t.x+go[i][0];
			y=t.y+go[i][1];
			if(1<=x&&x<=n&&1<=y&&y<=m&&min(dis[x][y],dis[t.x][t.y]+a[x][y])){
				g[&dis-f][x][y]=fr(&dis-f,t.x,t.y);
				if(!inq[x][y]){
					tail++;
					q[tail]=zt(x,y);
					inq[x][y]=1;
				}
			}
		}
	}
}
bool v[11][11];
void dfs(int s,int i,int j){
	v[i][j]=1;
	fr&f=g[s][i][j];
	if(f.hav()){
		dfs(f.s,f.x,f.y);
		if(f.x==i&&f.y==j)dfs(s^f.s,i,j);
	}
}
int main(){
	int i,j,s,t,K,inf;
	scanf("%d%d",&n,&m);
	memset(f,63,sizeof(f));
	inf=f[0][0][0];
	K=0;
	for(i=1;i<=n;i++){
		for(j=1;j<=m;j++){
			scanf("%d",a[i]+j);
			if(a[i][j]==0){
				f[1<<K][i][j]=0;
				K++;
			}
		}
	}
	for(s=0;s<1<<K;s++){
		head=1;
		tail=0;
		memset(inq,0,sizeof(inq));
		for(i=1;i<=n;i++){
			for(j=1;j<=m;j++){
				for(t=s;t;t=(t-1)&s){
					if(min(f[s][i][j],f[t][i][j]+f[s^t][i][j]-a[i][j]))g[s][i][j]=fr(t,i,j);
				}
				if(f[s][i][j]!=inf){
					tail++;
					q[tail]=zt(i,j);
					inq[i][j]=1;
				}
			}
		}
		bfs(f[s]);
	}
	for(i=1;i<=n;i++){
		for(j=1;j<=m;j++){
			if(a[i][j]==0){
				printf("%d\n",f[(1<<K)-1][i][j]);
				dfs((1<<K)-1,i,j);
				for(i=1;i<=n;i++){
					for(j=1;j<=m;j++){
						if(a[i][j]==0)
							putchar('x');
						else if(v[i][j])
							putchar('o');
						else
							putchar('_');
					}
					putchar('\n');
				}
				return 0;
			}
		}
	}
}

[BZOJ2595]游览计划的更多相关文章

  1. 【BZOJ2595】 [Wc2008]游览计划

    BZOJ2595 [Wc2008]游览计划 Solution 考虑这是一个最小费用连通性的问题,既然大家都说这是什么斯坦纳树那就是的吧... 所以我们肯定可以这样设一个dp状态: \(dp_{i,j, ...

  2. 【BZOJ2595】[Wc2008]游览计划 斯坦纳树

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

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

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

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

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

  5. bzoj2595 / P4294 [WC2008]游览计划

    P4294 [WC2008]游览计划 斯坦纳树 斯坦纳树,是一种神奇的树.它支持在一个连通图上求包含若干个选定点的最小生成树. 前置算法:spfa+状压dp+dfs(大雾) 我们设$f[o][P]$为 ...

  6. 【BZOJ-2595】游览计划 斯坦纳树

    2595: [Wc2008]游览计划 Time Limit: 10 Sec  Memory Limit: 256 MBSec  Special JudgeSubmit: 1518  Solved: 7 ...

  7. BZOJ_2595_[Wc2008]游览计划_斯坦纳树

    BZOJ_2595_[Wc2008]游览计划_斯坦纳树 题意: 分析: 斯坦纳树裸题,有几个需要注意的地方 给出矩阵,不用自己建图,但枚举子集转移时会算两遍,需要减去当前点的权值 方案记录比较麻烦,两 ...

  8. [WC2008]游览计划 解题报告

    [WC2008]游览计划 斯坦纳树板子题,其实就是状压dp 令\(dp_{i,s}\)表示任意点\(i\)联通关键点集合\(s\)的最小代价 然后有转移 \[ dp_{i,S}=\min_{T\in ...

  9. 【LG4294】[WC2008]游览计划

    [LG4294][WC2008]游览计划 题面 洛谷 bzoj 题解 斯坦纳树板子题. 斯坦纳树的总结先留个坑. 代码 #include <iostream> #include <c ...

随机推荐

  1. 微信小程序滑动选择器

    实现微信小程序滑动选择效果 在wxml文件中,用一个picker标签代表选择器,bindchange是用户点击确定后触发的函数,index是picker自带的参数,用户点击确定后,bindchange ...

  2. windows+nexus+maven环境搭建(转)

    windows nexus环境搭建 1.下载nexus 版本为 nexus-2.11.4-01-bundle 下载地址 这里写链接内容 2.将下载好的文件放到D盘进行解压 3.解压后目录结构 nexu ...

  3. python进行EDA探索性数据分析

    1.查看数据的类型概况 cols = [c for c in train.columns]   #返回数据的列名到列表里 print('Number of features: {}'.format(l ...

  4. exit() _exit() 函数区别

    exit(): --stdlib.h (1) 所有使用atexit()注册的函数,将会被以注册相反的顺序调用: (2) 所有打开的输出流被刷新,并且关闭流: (3) 使用tmpfile()创建的文件将 ...

  5. 64_k2

    kf5-libkdepim-akonadi-17.04.1-1.fc26.x86_64.rpm 25-May-2017 12:22 177198 kf5-libkdepim-devel-17.04.1 ...

  6. 菜鸟进阶之:VC++之Visual Studio中DLL调用实现

    C++写的DLL,用C++调用其实是一个比较简单的事情,调用DLL函数的方法其实有很多,说一个最普通的方法: 1.新建一个解决方案,文件->新建项目->Visual c++->win ...

  7. C基础 内存统一入口

    引言  - malloc 引述 C标准中堆上内存入口就只有 malloc, calloc, realloc . 内存回收口是 free. 常见的一种写法是 struct person * per = ...

  8. git学习笔记三

    1.每个分支的历史版本维护信息位置是.git/logs/refs/heads/master,这个位置的信息是文本文件,不是引用. harvey@harvey-Virtual-Machine:~/dem ...

  9. vue 开始开发

    1,引入vue.js文件 2,在body里用标签 编辑一个入口 <div id="app">{{msg}}</div> <-- 用双大括号 取数据显示 ...

  10. [转载] Python itertools模块详解

    原文在这里,写的很详细,感谢原作者,以下摘录要点. itertools用于高效循环的迭代函数集合. 无限迭代器 迭代器 参数 结果 例子 count() start, [step] start, st ...