上图论课的时候无意之间看到了这个,然后花了几天的时间学习了下,接下来做一个总结。

一般斯坦纳树问题是指(来自百度百科):

斯坦纳树问题是组合优化问题,与最小生成树相似,是最短网络的一种。最小生成树是在给定的点集和边中寻求最短网络使所有点连通。而最小斯坦纳树允许在给定点外增加额外的点,使生成的最短网络开销最小。

然后听说已经被证明为是NP问题了,在ACM竞赛中我们不研究这个,我们研究更简单一些的问题。

对于图G(V,E),其中V表示图中点集,E表示图中边集。设A是V的某个子集,求至少包含A中所有点的最小子树。

这个问题也是NP难问题。所以|A|必须很小才行。于是对于|A|很小的时候,有两种做法。

做法一: 根据性质暴力。

有一段很经典的做法(from here):

首先用floyd算法求出两两之间的最短路径。
然后把所有点都两两链接起来,权值就是它们的最短路径。
假设指定必须连接的点有K个。
那么MinimalSteinerTree 树中的内点最多有K-2个。
在纸上画一下就知道了,内点最多的情况就是树为满二叉树的情况。
而由于之前的floyd算法。把整个图给“缩”了一下。
所以树最多有K-2+K个点。
枚举所有可能的点集。对每个点集求最小生成树。取最小值即可。

如果这种方法是正确的,那么我们只需要在所有N(N=|A|)个点中选择K-2个点,然后进行最小生成树算法即可。复杂度为C(N,K-2)*K^2,对于N和K很小的时候还是可以的。

接下来就来证明,为什么做法一是正确的。

前提1:因为图中已经使用Floyd算法,所以图中不存在度为2的点

前提2:根节点全部为A集合中的点

推论1:内点的度数大于3

然后根据数据结构中一点树的基础知识我们可以得到公式:

设内点有X个

*(K+X-) >= K+*X

=》X <= K - 2

然后有同学可能会问,A集合中的点也可能不为根节点啊。稍微想下我们可以知道,如果存在Y个A集合中的点不为根节点,则对于公式左边没有影响,且必然会增大公式右边,所以使得X更小。

当时,这种思路虽然我觉得很好,但是这种方法在ACM中却都会超时。。。现在的出题人那里还会出纯的模板题。确实这方法局限性太大,学习学习思路就好了。

方法二:状态压缩DP

(学习了这个,了解到DP果真不是很适合我这种脑子不好用的玩家)

dp[mask][i],其中是以i为根,包含mask中的点的最小生成树的权值。(其实主要可以看成,这个树含有mask集合,且必有i这个点,mask集合是|A|的某个子集)

mask是点的集合,可以用二进制进行状态压缩。

在得知dp[mask-1 ~ 1][1...N]的情况下,如何推出dp[mask][1...N]呢?

两个步骤实现:

step1推出:

a = min{dp[m1][i] + dp[m2][i]},其中m1 | m2 = mask。 其中m1和m2就是mask的两个互补子集

step2推出

b = min(dp[mask][j] + d[j][i])

模板可以看我的这边博客。http://www.cnblogs.com/chenhuan001/p/4960239.html

接下来我简单来证明下这样做为什么是对的。

要得到dp[tmask][i] 也就是状态为tmask,且一定含有i这个点。 有两种情况

第一种,对于最优状态,i这个点为内点,那么一定可以找到tmask的两种互斥子集m1和m2,使得dp[tmask][i]=dp[m1][i]+dp[m2][i]

第二种,对于最优状态,i这个点为叶子节点,那么第一种方法已经无法找出,所有找到一个没有i节点的状态,添加一条指向i的边来进行转移。

这两种情况都考虑后,就能说明转移是正确的。

poj 3123

//
// main.cpp
// poj3123
//
// Created by 陈加寿 on 15/11/10.
// Copyright (c) 2015年 陈加寿. All rights reserved.
// #include <iostream>
#include <stdio.h>
#include <string.h>
#include <algorithm>
#include <string>
#include <map>
using namespace std;
#define N 33
#define INF 1000000000 map<string,int> Maphash;
int Maphashid; int mat[N][N];
int needconnect[][]; int cntcnt=;
int mi; void Init()
{
Maphash.clear();
Maphashid=;
for(int i=;i<N;i++)
for(int j=;j<N;j++)
if(i==j) mat[i][j]=;
else mat[i][j]=INF;
} //来一发spfa
int dp[][N];
int que[];
int qf,qd;
int spfamark[N]; int SteinerTreeDP(int mat[N][N],int maxid,int *sameset,int size,int* mark)
{
//所有的标点从1到maxid
//SteinerTree 所必须的点集为 sameset[0] - sameset[size-1]
memset(dp,,sizeof(dp));
for(int i=;i<(<<size);i++)
for(int j=;j<=maxid;j++)
dp[i][j]=INF;
//初始化都为INF for (int i=; i<size; i++) {
dp[(<<i)][sameset[i]]=;
} for (int i=;i<(<<size);i++)
{
int to[];
int tcnt=;
for(int j=;j<size;j++)
{
if( (i&(<<j)) != )
{
to[tcnt]=j;
tcnt++;
}
}
for(int j = ;j < (<<tcnt)- ;j++)//对于内部的每一种情况.
{
int tmp=;
for(int j1=;j1<tcnt;j1++)
{
if( ((<<j1)&j) != )
{
tmp |= (<<to[j1]);
}
}
int othertmp= (~tmp)&i;
if( (tmp|othertmp) != i) printf("error! tmp|other != i\n"),needconnect[][];
for(int k=;k<=maxid;k++)
dp[i][k]=min(dp[i][k],dp[tmp][k]+dp[othertmp][k]);//是含有,并不是只含有!
}
//然后开始SPFA
//一开始把所有不为INF的点入队列
qf=qd=;
memset(spfamark, , sizeof(spfamark));
for(int j=;j<=maxid;j++)
{
if(dp[i][j]!=INF)
{
que[qf++]=j;
spfamark[j]=;
}
}
while(qf>qd)
{
int cur=que[qd++];
spfamark[cur]=;
for(int j=;j<=maxid;j++)
{
if( mat[cur][j]!=INF && j!=cur && dp[i][j] > dp[i][cur]+mat[cur][j] )
{
dp[i][j] = dp[i][cur] + mat[cur][j];
if(spfamark[j]==)
{
spfamark[j]=;
que[qf++]=j;
}
}
}
} }
int tmin=INF;
for(int i=;i<=maxid;i++)
tmin=min(tmin,dp[(<<size)-][i]);
return tmin;
} int getmin(int *sameset,int size)
{
if(size>) return ;
if(size<) return ;
//两个的时候不要抗拒
// if(size==2)
// {
// return mat[ sameset[0] ][ sameset[1] ];
// }
int save[N];
int cnt=;
for(int i=;i<=Maphashid;i++)
{
int sign=;
for(int j=;j<size;j++)
{
if(sameset[j]==i) {
sign=;
break;
}
}
if(sign==)
save[cnt++]=i;
}
//然后dfs找size-2种可能。
int mark[N];
memset(mark,,sizeof(mark));
for(int i=;i<size;i++)
mark[ sameset[i] ]=;
mi=INF;
return SteinerTreeDP(mat,Maphashid,sameset,size,mark);
} int main(int argc, const char * argv[]) {
// insert code here...
int n,m;
while (scanf("%d%d",&n,&m)&&(n+m)) {
Init();
for(int i=;i<n;i++)
{
string tmp;
cin>>tmp;
if(Maphash[tmp]==)
Maphash[tmp] = ++Maphashid;
}
for(int i=;i<m;i++)
{
string a,b;
int hasha,hashb,len;
cin>>a>>b>>len;
hasha=Maphash[a];
hashb=Maphash[b]; mat[hasha][hashb]=mat[hashb][hasha]=min(mat[hasha][hashb],len);
}
for(int i=;i<;i++)
{
//四个
string a,b;
cin>>a>>b;
needconnect[i][]=Maphash[a];
needconnect[i][]=Maphash[b];
} //floyd 还没有用 for(int i=;i<=Maphashid;i++)
for(int j=;j<=Maphashid;j++)
for(int k=;k<=Maphashid;k++)
if(mat[i][j] > mat[i][k]+mat[k][j])
{
mat[i][j] = mat[i][k]+mat[k][j];
} int sameset[];
int setnum;
int ans=INF;
for(int i=;i<=;i++)
for(int i1=;i1<=i+;i1++)
for(int i2=;i2<=i1+;i2++)
for(int i3=;i3<=i2+;i3++)
{
//if(i1==0&&i2==0&&i3==0&&i==0) continue;//这个减掉了竟然还超时 我就抄!
int sum=;
for(int j=;j<;j++)
{
//取出同一集合
setnum=;
if(i==j)
{
sameset[setnum++]=needconnect[][];
sameset[setnum++]=needconnect[][];
}
if(i1==j)
{
sameset[setnum++]=needconnect[][];
sameset[setnum++]=needconnect[][];
}
if(i2==j)
{
sameset[setnum++]=needconnect[][];
sameset[setnum++]=needconnect[][];
}
if(i3==j)
{
sameset[setnum++]=needconnect[][];
sameset[setnum++]=needconnect[][];
}
if(setnum==||setnum==) continue;
//去重复
sort(sameset,sameset+setnum);
int tsetnum=;
for(int k=;k<setnum;k++)
{
if(sameset[k] != sameset[k-])
{
sameset[tsetnum]=sameset[k];
tsetnum++;
}
} sum += getmin(sameset,tsetnum);
}
ans=min(ans,sum);
}
printf("%d\n",ans);
//printf("cntcnt: %d\n",cntcnt); }
return ;
}

zoj 3613

//
// main.cpp
// zoj3613
//
// Created by 陈加寿 on 15/11/12.
// Copyright (c) 2015年 陈加寿. All rights reserved.
// #include <iostream>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <algorithm>
using namespace std;
#define N 202
#define INF 100000000
#define K 8 int R[];
int F[];
int palne[N];
int flagr[N];
int mat[N][N];
int cntr=;
int cntf=;
int maxnum=;
int micost=; //SteinerTree 邻接矩阵模板。(稠密图)时间复杂度 O(N*2^K*(2^K+N))
int dp[(<<K)+][N];
int midp[(<<K)+];
int STV[N]; int SteinerTreeDP(int mat[N][N],int maxid,int *sameset,int size)
{
//mat为表示距离的邻接矩阵
//所有的标点从1到maxid
//SteinerTree 所必须的点集为 sameset[0] 到 sameset[size-1]
//函数放回最小Steiner Tree的值 for(int i=;i<(<<size);i++)
for(int j=;j<=maxid;j++)
dp[i][j]=INF; for (int i=; i<size; i++) {
dp[(<<i)][sameset[i]]=;
}
for (int i=;i<(<<size);i++)
{
//step 1
for(int kk=;kk<=maxid;kk++)
{
STV[kk]=;
for(int j = (i-)&i ; j ;j = (j-)&i)
{
dp[i][kk] = min(dp[i][kk],dp[j][kk]+dp[(~j)&i][kk]);
}
}
//step 2
int kk,stmin=INF,stminid=;
for (int j = ; stmin = INF, j < maxid; j++)
{
for (kk = ; kk <= maxid; kk++)
if (dp[i][kk] <= stmin && !STV[kk])
stmin = dp[i][stminid = kk]; for (STV[stminid]=,kk = ; kk <= maxid; kk++)
if(STV[kk]==) dp[i][kk] = min(dp[i][kk], dp[i][stminid] + mat[stminid][kk]);
}
} int tmin=INF;
for(int j=;j<=maxid;j++)
tmin=min(tmin,dp[(<<size)-][j]);
return tmin;
} int check(int s)
{
int cnt=;
int i;
for(i=;i<cntf;i++)
if( ((<<i)&s)!= ) cnt += palne[ F[i] ];
for(int j=;j<cntr;j++)
if( ((<<(cntf+j))&s) != ) cnt --;
if( cnt>= ) return ;
return ;
} int getwei(int s)
{
int cnt=;
for(int j=;j<cntr;j++)
{
if( ((<<(cntf+j))&s) != ) cnt ++;
}
return cnt;
} int main(int argc, const char * argv[]) {
int n;
while(scanf( "%d",&n )!=EOF)
{
maxnum=;
micost=;
cntr=;
cntf=;
int saveans=;
for(int i=;i<=n;i++)
{
scanf("%d%d",&palne[i],&flagr[i]);
if(flagr[i]==&&palne[i]!=)
{
saveans++;
flagr[i]=;
palne[i]--;
//自厂自销
}
if(flagr[i]==)
{
R[cntr++]=i;
}
if(palne[i]!=)
{
F[cntf++]=i;
}
}
int m;
scanf("%d",&m);
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
{
if(i==j) mat[i][j]=;
else mat[i][j]=INF;
} for(int i=;i<m;i++)
{
int x,y,w;
scanf("%d%d%d",&x,&y,&w);
mat[x][y]=mat[y][x]=min(mat[x][y],w);
}
int sameset[];
int size=;
for(int i=;i<cntf;i++)
sameset[ size++ ] = F[i];
for(int i=;i<cntr;i++)
sameset[ size++ ] = R[i];
//等下在这里再写一个用连接表的。
SteinerTreeDP(mat, n, sameset, size); int mask=((<<size)-);
for(int i=;i<=mask;i++)
{
midp[i]=INF;
for(int j=;j<=n;j++)
midp[i]=min(midp[i] , dp[i][j]);
}
//然后最后的一个DP
for(int i=;i<=mask;i++)
{
//第一开始要满足要求
if( check(i)== ) continue;
for(int j=(i-)&i;j;j=(j-)&i)
{
if( check(j)&&check(i-j) ) midp[i]=min(midp[i],midp[j]+midp[i-j]);
}
int tnum=getwei(i);
if(tnum>=maxnum)
{
if(tnum>maxnum)
{
maxnum=tnum;
micost=midp[i];
}
else if(midp[i] < micost)
{
micost=midp[i];
}
}
} printf("%d %d\n",maxnum+saveans,micost);
}
return ;
}

hdu4085和zoj 3613很类似,记住最后再来个DP就可以省很多代码量。

Minimal Steiner Tree ACM的更多相关文章

  1. HDU 2489 Minimal Ratio Tree (dfs+Prim最小生成树)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2489 Problem Description For a tree, which nodes and ...

  2. hdu2489 Minimal Ratio Tree

    hdu2489 Minimal Ratio Tree 题意:一个 至多  n=15 的 完全图 ,求 含有 m 个节点的树 使 边权和 除 点权和 最小 题解:枚举 m 个 点 ,然后 求 最小生成树 ...

  3. HDU 2489 Minimal Ratio Tree (DFS枚举+最小生成树Prim)

    Minimal Ratio Tree Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 32768/32768K (Java/Other) ...

  4. HDU 2489 Minimal Ratio Tree 最小生成树+DFS

    Minimal Ratio Tree Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Other ...

  5. HDUOJ----2489 Minimal Ratio Tree

    Minimal Ratio Tree Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Other ...

  6. HDU 2489 Minimal Ratio Tree(暴力+最小生成树)(2008 Asia Regional Beijing)

    Description For a tree, which nodes and edges are all weighted, the ratio of it is calculated accord ...

  7. HDU2489 Minimal Ratio Tree 【DFS】+【最小生成树Prim】

    Minimal Ratio Tree Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Other ...

  8. Minimal Ratio Tree HDU - 2489

    Minimal Ratio Tree HDU - 2489 暴力枚举点,然后跑最小生成树得到这些点时的最小边权之和. 由于枚举的时候本来就是按照字典序的,不需要额外判. 错误原因:要求输出的结尾不能有 ...

  9. HDU 2489 Minimal Ratio Tree(prim+DFS)

    Minimal Ratio Tree Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Other ...

随机推荐

  1. Intellij IDEA 2017 破解

    http://idea.lanyus.com/ https://www.cnblogs.com/wang1024/p/7485758.html

  2. perl学习笔记——目录操作

    在目录书中移动 chdir 操作副改变当前的工作目录.它和shell中cd命令类似: chdir ‘/etc’ or die “cannot chdir to /etc:$1”; 注意:工作目录不能更 ...

  3. Rails生成随机字符串及加解密

    1.生成安全的base64字符串 key = SecureRandom.urlsafe_base64

  4. 【Android】The application has stopped unexpectedly.Please try again.

    打出的android APK在android4.0.2手机上运行正常,但是在2.3.7的MOTO DEFY ME525上运行失败. android 2.3.3版本的模拟器上面运行失败: Sorry! ...

  5. HDU - 3038 How Many Answers Are Wrong (带权并查集)

    题意:n个数,m次询问,每次问区间a到b之间的和为s,问有几次冲突 思路:带权并查集的应用.[a, b]和为s,所以a-1与b就能够确定一次关系.通过计算与根的距离能够推断出询问的正确性 #inclu ...

  6. libevent2源码分析之五:关键的调用链

    用一个调用链来表示函数调用的流程,看起来更直观.根据上面的分析,总结了一些重要的调用链. 初始化 event_base_new event_base_new_with_config min_heap_ ...

  7. HTML5特性

    HTML5规范围绕如何使用新增标记定义了大量Javascript API,其中一些API与DOM重叠,定义了浏览器应该支持DOM拓展. 1.与类相关的扩充 HTML5新增了getElementsByC ...

  8. C# 6.0语法糖剖析

    C# 6.0语法糖剖析 2016年12月16日 16:10:27 阅读数:586   版权声明:本文为博主http://www.feixueteam.net原创文章,未经博主允许不得转载. https ...

  9. linux 实时显示文件的内容

    1. watch -n 1 aa.txt  #每个1秒显示aa.txt的内容 2. tail -f ***.log Linux shell中有一个tail命令,常用来显示一个文件的最后n行文档内容 但 ...

  10. unity Changing Game View background color

    Change the background color in the camera 参考:http://forum.unity3d.com/threads/changing-game-view-bac ...