bzoj2595: [Wc2008]游览计划 斯坦纳树
斯坦纳树是在一个图中选取某些特定点使其联通(可以选取额外的点),要求花费最小,最小生成树是斯坦纳树的一种特殊情况
我们用dp[i][j]来表示以i为根,和j状态是否和i联通,那么有
转移方程:
dp[i][j]=min(dp[i][s]+dp[j-s]-a[i][j]) (表示有两个状态s和j-s都和i联通,我们把这两个状态联通起来,这样多算了一次a[i][j],减去即可)
dp[i][j]=min(dp[i][j],dp[k][j]+a[i][k]) (如果i和k连着,那么,链接i和k,更新dp[i][j]),此处类似与最短路中的松弛操作
来看这道题就是求平面上给定点的斯坦纳树,要求输出路径,我们更新状态的时候记录pre,
spfa松弛(164 ms):
/**************************************************************
Problem: 2595
User: walfy
Language: C++
Result: Accepted
Time:164 ms
Memory:7756 kb
****************************************************************/
//#pragma comment(linker, "/stack:200000000")
//#pragma GCC optimize("Ofast,no-stack-protector")
//#pragma GCC target("sse,sse2,sse3,ssse3,sse4,popcnt,abm,mmx,avx,tune=native")
//#pragma GCC optimize("unroll-loops")
#include<bits/stdc++.h>
#define fi first
#define se second
#define mp make_pair
#define pb push_back
#define pi acos(-1.0)
#define ll long long
#define vi vector<int>
#define mod 1000000007
#define C 0.5772156649
#define ls l,m,rt<<1
#define rs m+1,r,rt<<1|1
#define pil pair<int,ll>
#define pli pair<ll,int>
#define pii pair<int,int>
#define cd complex<double>
#define ull unsigned long long
#define base 1000000000000000000
#define Max(a, b) ((a)>(b)?(a):(b))
#define Min(a, b) ((a)<(b)?(a):(b))
#define fio ios::sync_with_stdio(false);cin.tie(0)
inline void add(ll &a,ll b){a+=b;if(a>=mod)a-=mod;}
inline ll gcd(ll a,ll b){return b?gcd(b,a%b):a;}
inline ll qp(ll a,ll b){ll ans=1;while(b){if(b&1)ans=ans*a%mod;a=a*a%mod,b>>=1;}return ans;}
using namespace std;
const double g=10.0,eps=1e-12;
const int N=10+10,maxn=(1<<10)+10,inf=0x3f3f3f3f,INF=0x3f3f3f3f3f3f3f3f;
struct Pre{
int x,y,st;
}pre[N][N][maxn];
int f[N][N][maxn],n,m,a[N][N];
queue<pii>q;
bool vis[N][N];
int dx[4]={0,0,1,-1};
int dy[4]={1,-1,0,0};
void spfa(int st)
{
while(!q.empty())
{
pii p=q.front();q.pop();
vis[p.fi][p.se]=0;
for(int i=0;i<4;i++)
{
int nx=p.fi+dx[i],ny=p.se+dy[i];
if(1<=nx&&nx<=n&&1<=ny&&ny<=m)
{
if(f[p.fi][p.se][st]+a[nx][ny]<f[nx][ny][st])
{
f[nx][ny][st]=f[p.fi][p.se][st]+a[nx][ny];
pre[nx][ny][st]={p.fi,p.se,st};
if(!vis[nx][ny])
{
vis[nx][ny]=1;
q.push(mp(nx,ny));
}
}
}
}
}
}
void dfs(int x,int y,int st)
{
vis[x][y]=1;
Pre p=pre[x][y][st];
if(!p.x)return ;
dfs(p.x,p.y,p.st);
if(x==p.x&&y==p.y)
dfs(p.x,p.y,st-p.st);
}
int main()
{
int cnt=0,ansx=-1,ansy;
scanf("%d%d",&n,&m);
memset(f,inf,sizeof f);
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
scanf("%d",&a[i][j]);
if(!a[i][j])
{
f[i][j][(1<<cnt)]=0;cnt++;
if(ansx==-1)ansx=i,ansy=j;
}
}
}
for(int st=0;st<(1<<cnt);st++)
{
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
for(int s=st;s;s=(s-1)&st)
{
if(f[i][j][s]+f[i][j][st-s]-a[i][j]<f[i][j][st])
{
f[i][j][st]=f[i][j][s]+f[i][j][st-s]-a[i][j];
pre[i][j][st]={i,j,s};
}
}
if(f[i][j][st]<inf)q.push(mp(i,j)),vis[i][j]=1;
}
}
spfa(st);
}
printf("%d\n",f[ansx][ansy][(1<<cnt)-1]);
memset(vis,0,sizeof vis);
dfs(ansx,ansy,(1<<cnt)-1);
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
if(!a[i][j])putchar('x');
else putchar(vis[i][j]?'o':'_');
}
puts("");
}
return 0;
}
/***********************
***********************/
dij松弛(432 ms):
/**************************************************************
Problem: 2595
User: walfy
Language: C++
Result: Accepted
Time:432 ms
Memory:7756 kb
****************************************************************/
//#pragma comment(linker, "/stack:200000000")
//#pragma GCC optimize("Ofast,no-stack-protector")
//#pragma GCC target("sse,sse2,sse3,ssse3,sse4,popcnt,abm,mmx,avx,tune=native")
//#pragma GCC optimize("unroll-loops")
#include<bits/stdc++.h>
#define fi first
#define se second
#define mp make_pair
#define pb push_back
#define pi acos(-1.0)
#define ll long long
#define vi vector<int>
#define mod 1000000007
#define C 0.5772156649
#define ls l,m,rt<<1
#define rs m+1,r,rt<<1|1
#define pil pair<int,ll>
#define pli pair<ll,int>
#define pii pair<int,int>
#define cd complex<double>
#define ull unsigned long long
#define base 1000000000000000000
#define Max(a, b) ((a)>(b)?(a):(b))
#define Min(a, b) ((a)<(b)?(a):(b))
#define fio ios::sync_with_stdio(false);cin.tie(0)
inline void add(ll &a,ll b){a+=b;if(a>=mod)a-=mod;}
inline ll gcd(ll a,ll b){return b?gcd(b,a%b):a;}
inline ll qp(ll a,ll b){ll ans=1;while(b){if(b&1)ans=ans*a%mod;a=a*a%mod,b>>=1;}return ans;}
using namespace std;
const double g=10.0,eps=1e-12;
const int N=10+10,maxn=(1<<10)+10,inf=0x3f3f3f3f,INF=0x3f3f3f3f3f3f3f3f;
struct Pre{
int x,y,st;
}pre[N][N][maxn];
int f[N][N][maxn],n,m,a[N][N];
bool vis[N][N];
int dx[4]={0,0,1,-1};
int dy[4]={1,-1,0,0};
int state;
struct node{
int x,y;
bool operator <(const node&rhs)const{
return f[x][y][state]<f[rhs.x][rhs.y][state];
}
};
priority_queue<node>q;
void dij(int st)
{
while(!q.empty())
{
node p=q.top();q.pop();
for(int i=0;i<4;i++)
{
int nx=p.x+dx[i],ny=p.y+dy[i];
if(1<=nx&&nx<=n&&1<=ny&&ny<=m)
{
if(f[p.x][p.y][st]+a[nx][ny]<f[nx][ny][st])
{
f[nx][ny][st]=f[p.x][p.y][st]+a[nx][ny];
pre[nx][ny][st]={p.x,p.y,st};
q.push({nx,ny});
}
}
}
}
}
void dfs(int x,int y,int st)
{
vis[x][y]=1;
Pre p=pre[x][y][st];
if(!p.x)return ;
dfs(p.x,p.y,p.st);
if(x==p.x&&y==p.y)
dfs(p.x,p.y,st-p.st);
}
int main()
{
int cnt=0,ansx=-1,ansy;
scanf("%d%d",&n,&m);
memset(f,inf,sizeof f);
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
scanf("%d",&a[i][j]);
if(!a[i][j])
{
f[i][j][(1<<cnt)]=0;cnt++;
if(ansx==-1)ansx=i,ansy=j;
}
}
}
for(int st=0;st<(1<<cnt);st++)
{
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
for(int s=st;s;s=(s-1)&st)
{
if(f[i][j][s]+f[i][j][st-s]-a[i][j]<f[i][j][st])
{
f[i][j][st]=f[i][j][s]+f[i][j][st-s]-a[i][j];
pre[i][j][st]={i,j,s};
}
}
if(f[i][j][st]<inf)q.push({i,j});
}
}
state=st;
dij(st);
}
printf("%d\n",f[ansx][ansy][(1<<cnt)-1]);
memset(vis,0,sizeof vis);
dfs(ansx,ansy,(1<<cnt)-1);
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
if(!a[i][j])putchar('x');
else putchar(vis[i][j]?'o':'_');
}
puts("");
}
return 0;
}
/***********************
***********************/
bzoj2595: [Wc2008]游览计划 斯坦纳树的更多相关文章
- 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][ ...
- 【BZOJ2595】[Wc2008]游览计划 斯坦纳树
[BZOJ2595][Wc2008]游览计划 Description Input 第一行有两个整数,N和 M,描述方块的数目. 接下来 N行, 每行有 M 个非负整数, 如果该整数为 0, 则该方块为 ...
- Luogu 4294 [WC2008]游览计划 | 斯坦纳树
题目链接 Luogu 4294 (我做这道题的时候BZOJ全站的SPJ都炸了 提交秒WA 幸好有洛谷) 题解 这道题是[斯坦纳树]的经典例题.斯坦纳树是这样一类问题:带边权无向图上有几个(一般约10个 ...
- 【BZOJ-2595】游览计划 斯坦纳树
2595: [Wc2008]游览计划 Time Limit: 10 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 1518 Solved: 7 ...
- BZOJ 2595 [Wc2008]游览计划 ——斯坦纳树
[题目分析] 斯坦纳树=子集DP+SPFA? 用来学习斯坦纳树的模板. 大概就是用二进制来表示树包含的点,然后用跟几点表示树的形态. 更新分为两种,一种是合并两个子集,一种是换根,换根用SPFA迭代即 ...
- P4294 [WC2008]游览计划 (斯坦纳树)
题目链接 差不多是斯坦纳树裸题,不过边权化成了点权,这样在合并两棵子树时需要去掉根结点的权值,防止重复. 题目还要求输出解,只要在转移时记录下路径,然后dfs一遍就好了. #include<bi ...
- 洛谷4294 [WC2008]游览计划——斯坦纳树
题目:https://www.luogu.org/problemnew/show/P4294 大概是状压.两种转移,一个是以同一个点为中心,S由自己的子集拼起来:一个是S相同.中心不同的同层转移. 注 ...
- BZOJ2595 Wc2008 游览计划 【斯坦纳树】【状压DP】*
BZOJ2595 Wc2008 游览计划 Description Input 第一行有两个整数,N和 M,描述方块的数目. 接下来 N行, 每行有 M 个非负整数, 如果该整数为 0, 则该方块为一个 ...
随机推荐
- 如何合并两个Git仓库
欢迎和大家交流技术相关问题: 邮箱: jiangxinnju@163.com 博客园地址: http://www.cnblogs.com/jiangxinnju GitHub地址: https://g ...
- Linux学习笔记之Centos7安装GNOME桌面环境
最小化安装Centos7,系统默认是命令行界面,如果像我一样有特殊需求,这时就需要我们手动来安装用户图形界面了. 1.查看一下当前的运行级别和可以安装的group. systemctl get-def ...
- 20145330 《网络对抗》 Web安全基础实践
20145330 <网络对抗> Web安全基础实践 1.实验后回答问题 (1)SQL注入攻击原理,如何防御 SQL注入,就是通过把SQL命令插入到Web表单提交或输入域名或页面请求的查询字 ...
- 20165310 Java实验五《网络编程与安全》
20165310 Java实验五<网络编程与安全> 任务一 题目:①编写MyBC.java实现中缀表达式转后缀表达式的功能:②编写MyDC.java实现从上面功能中获取的表达式中实现后缀表 ...
- Python3基础 list str转成list
Python : 3.7.0 OS : Ubuntu 18.04.1 LTS IDE : PyCharm 2018.2.4 Conda ...
- ExtJS使用入门
extjs是基于 yui 由 jack slocum开发, sencha是他们的公司, sencha是由三个项目合并起来的开源项目: ExtJS, jqTouch, Raphael(拉斐尔, 圣经中的 ...
- 51NOD 1027 大数乘法
1027 大数乘法 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 给出2个大整数A,B,计算A*B的结果. Input 第1行:大数A 第2行:大数B (A,B ...
- [不屈的复习] - 安装Java初始化环境
点WIN键->运行(或者使用win+r) 输入cmd命令输入java -version 注: -version是小写,不能使用大写,java后面有一个空格 配置成功后,会出现版本信息 java ...
- urljoin
from urlparse import urljoin urljoin("http://www.asite.com/folder/currentpage.html", " ...
- FAILED Selenium2Library
FAILED Selenium2Library Importing test library 'Selenium2Library' failed: ImportError: cannot import ...