BZOJ 2595 [Wc2008]游览计划 ——斯坦纳树
【题目分析】
斯坦纳树=子集DP+SPFA?
用来学习斯坦纳树的模板。
大概就是用二进制来表示树包含的点,然后用跟几点表示树的形态。
更新分为两种,一种是合并两个子集,一种是换根,换根用SPFA迭代即可。
【代码】
- #include <cstdio>
- #include <cstring>
- #include <cstdlib>
- #include <cmath>
- #include <set>
- #include <map>
- #include <string>
- #include <algorithm>
- #include <vector>
- #include <iostream>
- #include <queue>
- using namespace std;
- #define maxn 11
- #define F(i,j,k) for (int i=j;i<=k;++i)
- #define inf (0x3f3f3f3f)
- int Getint()
- {
- int x=0,f=1; char ch=getchar();
- while (ch<'0'||ch>'9') {if (ch=='-') f=-1; ch=getchar();}
- while (ch>='0'&&ch<='9') {x=x*10+ch-'0'; ch=getchar();}
- return x*f;
- }
- int n,m,a[maxn][maxn],cnt=0,dp[maxn][maxn][1<<maxn],pre[maxn][maxn][1<<maxn][4];
- queue <int> qi,qj;
- int inq[maxn][maxn],b[maxn][maxn];
- int mov[4][2]={0,1,1,0,-1,0,0,-1};
- void dfs(int x,int y,int k)
- {
- // cout<<"dfs "<<x<<" "<<y<<" "<<k<<endl;
- // getchar();
- if (!k||!x||!y) return;
- b[x][y]=1;
- dfs(pre[x][y][k][0],pre[x][y][k][1],pre[x][y][k][2]);
- if (pre[x][y][k][2]!=k) dfs(pre[x][y][k][0],pre[x][y][k][1],k^pre[x][y][k][2]);
- // if (pre[x][y][k][0]==x&&pre[x][y][k][1]==y) dfs(pre[x][y][k][0],pre[x][y][k][1],k^pre[x][y][k][2]);
- }
- void out()
- {
- F(i,1,n)
- {
- F(j,1,m)
- if (b[i][j])
- {
- if (a[i][j]) printf("o");
- else printf("x");
- }
- else
- {
- printf("_");
- }
- printf("\n");
- }
- }
- int main()
- {
- memset(dp,0x3f,sizeof dp);
- n=Getint();m=Getint();
- F(i,1,n) F(j,1,m)
- {
- a[i][j]=Getint();
- if(!a[i][j])
- {
- cnt++;
- dp[i][j][1<<(cnt-1)]=0;
- }
- }
- for (int k=1;k<(1<<cnt);++k)
- {
- F(i,1,n) F(j,1,m)
- {
- for (int x=(k-1)&k;x;x=(x-1)&k)
- {
- int tmp=dp[i][j][x]+dp[i][j][k^x]-a[i][j];
- if (tmp<dp[i][j][k])
- {
- dp[i][j][k]=tmp;
- pre[i][j][k][0]=i;
- pre[i][j][k][1]=j;
- pre[i][j][k][2]=x;
- }
- }
- if (dp[i][j][k]<inf)
- {
- qi.push(i);
- qj.push(j);
- inq[i][j]=1;
- }
- }
- while (!qi.empty())
- {
- int ni=qi.front(),nj=qj.front();
- qi.pop();qj.pop();
- inq[ni][nj]=0;
- F(i,0,3)
- {
- int ti=ni+mov[i][0],tj=nj+mov[i][1];
- if (ti<=0||ti>n||tj<=0||tj>m) continue;
- if (dp[ni][nj][k]+a[ti][tj]<dp[ti][tj][k])
- {
- dp[ti][tj][k]=dp[ni][nj][k]+a[ti][tj];
- pre[ti][tj][k][0]=ni;
- pre[ti][tj][k][1]=nj;
- pre[ti][tj][k][2]=k;
- if (!inq[ti][tj])
- {
- qi.push(ti);
- qj.push(tj);
- inq[ti][tj]=1;
- }
- }
- }
- }
- }
- F(i,1,n) F(j,1,m)
- {
- if (!a[i][j])
- {
- printf("%d\n",dp[i][j][(1<<cnt)-1]);
- dfs(i,j,(1<<cnt)-1);
- out();
- return 0;
- }
- }
- }
BZOJ 2595 [Wc2008]游览计划 ——斯坦纳树的更多相关文章
- 【BZOJ2595】[Wc2008]游览计划 斯坦纳树
[BZOJ2595][Wc2008]游览计划 Description Input 第一行有两个整数,N和 M,描述方块的数目. 接下来 N行, 每行有 M 个非负整数, 如果该整数为 0, 则该方块为 ...
- Luogu 4294 [WC2008]游览计划 | 斯坦纳树
题目链接 Luogu 4294 (我做这道题的时候BZOJ全站的SPJ都炸了 提交秒WA 幸好有洛谷) 题解 这道题是[斯坦纳树]的经典例题.斯坦纳树是这样一类问题:带边权无向图上有几个(一般约10个 ...
- bzoj2595: [Wc2008]游览计划 斯坦纳树
斯坦纳树是在一个图中选取某些特定点使其联通(可以选取额外的点),要求花费最小,最小生成树是斯坦纳树的一种特殊情况 我们用dp[i][j]来表示以i为根,和j状态是否和i联通,那么有 转移方程: dp[ ...
- bzoj2595 [Wc2008]游览计划——斯坦纳树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2595 今天刚学了斯坦纳树,还不太会,写一道题练习一下: 参考了博客:http://www.c ...
- BZOJ2595: [Wc2008]游览计划(斯坦纳树,状压DP)
Time Limit: 10 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 2030 Solved: 986[Submit][Status][ ...
- P4294 [WC2008]游览计划 (斯坦纳树)
题目链接 差不多是斯坦纳树裸题,不过边权化成了点权,这样在合并两棵子树时需要去掉根结点的权值,防止重复. 题目还要求输出解,只要在转移时记录下路径,然后dfs一遍就好了. #include<bi ...
- 洛谷4294 [WC2008]游览计划——斯坦纳树
题目:https://www.luogu.org/problemnew/show/P4294 大概是状压.两种转移,一个是以同一个点为中心,S由自己的子集拼起来:一个是S相同.中心不同的同层转移. 注 ...
- 【BZOJ-2595】游览计划 斯坦纳树
2595: [Wc2008]游览计划 Time Limit: 10 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 1518 Solved: 7 ...
- bzoj 2595 [Wc2008]游览计划(斯坦纳树)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2595 [题意] 给定N*M的长方形,选最少权值和的格子使得要求的K个点连通. [科普] ...
随机推荐
- windows下jdk环境变量配置
JAVA_HOMEC:\Program Files\Java\jdk1.8.0_131 JMETER_HOMEC:\jmeter\jmeter3.2 CLASSPATH%JAVA_HOME%\lib; ...
- HDU 5469 Antonidas (树形DP,暴力)
题意: 给一棵n节点的树图,每个点都是一个小写字母,要求找到两个点(a,b),从a->b的路径上形成了一个字符串为s.给出s,问是否存在这样的点对. 思路: 考虑一个点,要么从该点出发,要么在该 ...
- DRM 简介
首先,我们对和DRM 相关的一些概念进行介绍. Buffer: 对于RAC 数据库,当一个数据块被读入到buffer cache后,我们就称其为buffer , cache fusion 会将这个bu ...
- 利用enum4linux 445端口+wordpress插件任意文件上传的一次渗透
探测内网80端口发现目标IP 目标使用Apache 2.4.7web服务中间件 使用linux Ubuntu系统 使用御剑扫描了目录 目录扫描到了 http://192.168.31.236/ ...
- 通过例子理解 k8s 架构【转】
为了帮助大家更好地理解 Kubernetes 架构,我们部署一个应用来演示各个组件之间是如何协作的. 执行命令 kubectl run httpd-app --image=httpd --replic ...
- 计算机图形学(Conputer Graphics):非均匀有理B样条
计算机图形学(Conputer Graphics):非均匀有理B样条 非均匀有理B样条(Non-Uniform Rational B-Spline)英文缩写,NURBS. 它是贝塞尔曲线的一个推广,而 ...
- linux下怎么修改mysql的字符集编码
安装完的MySQL的默认字符集为 latin1 ,为了要将其字符集改为用户所需要的(比如utf8),就必须改其相关的配置文件:由于linux下MySQL的默认安装目录分布在不同的文件下:不像windo ...
- kubernetes概念
kubernetes blog cluster cluster是计算.存储.和网络资源的集合,kubernetes利用这些资源运行各种基于容器的应用. master master是cluster的大脑 ...
- javascript顺序数组简单实现个二分查找
直接上码了注释写得很详细: function bsearch(A,x){ //l:查找范围左 r:查找范围右 let l = 0, //查询范围左边界 r = A.length-1, //查找范围右边 ...
- Git学习——工作区和暂存区
工作区就是我们的电脑上的git初始化目录.版本库就是我们工作区中的隐藏目录.git.版本库中分为两个部分:(1)stage(index)暂存区:git add <file>命令后file就 ...