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, 则该方块为一个 ...
随机推荐
- JVM内存分布
参考引用文章地址: http://hllvm.group.iteye.com/group/wiki/3053-JVMhttp://blog.csdn.net/william001zs/article/ ...
- Core Java 2
p267~p270: 1.一个方法不仅需要告诉编译器将要返回什么值, 还要告诉编译器有可能发生什么错误(以便在错误发生时用妥善的方式处理错误). 2.方法应该在首部声明所有可能抛出的异常. 3.方法抛 ...
- Python Web学习笔记之SSL,TLS,HTTPS
一. SSL 1. SSL简介 SSL协议位于TCP/IP协议与各种应用层协议之间,为数据通讯提供安全支持.SSL协议可分为两层: SSL记录协议(SSL Record Protocol):它建立在可 ...
- 02: 安装epel 解决centos7无法使用yum安装nginx
参考网址: http://www.mamicode.com/info-detail-1671603.html 1.yum命令安装 yum install epel-release -y 2.更新数据 ...
- 关于mysql连接抛出10038错误问题
今天用Navicat Premium连接windows server 2003 mysql的时候, 抛出10038问题, 这种问题之前在rhel也出现过一次, 就是防火墙不允许连接kill掉了这个请求 ...
- 20145106 《Java程序设计》第8周学习总结
教材学习内容总结 NIO即New IO.对于高级输入/输出处理.java从JDK1.4开始提供了NIO,在JAVA SE 7 中又提供了NIO2,认识这些高级输入/输出处理API(Applicatio ...
- tf.truncated_normal的用法
tf.truncated_normal(shape, mean, stddev) :shape表示生成张量的维度,mean是均值,stddev是标准差.这个函数产生正太分布,均值和标准差自己设定.这是 ...
- python数据库编程小应用
python DB api 数据库连接对象connection数据库交互对象cursor数据库异常类exceptions 流程:开始创建connection获取cursor执行查询.执行命令.获取数据 ...
- ubuntu下如何修改时区和时间
1.修改时区 sudo tzselect (按提示选择即可) sudo cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime 2. 修改时间 sudo ...
- 分布式系统一致性协议--Paxos算法
Paxos: Paxos算法背景介绍: Paxos算法是分布式技术大师Lamport提出的,主要目的是通过这个算法,让参与分布式处理的每个参与者逐步达成一致意见.用好理解的方式来说,就是在一个选举过程 ...