2019.03.30 洛谷 P4784 [BalticOI 2016 Day2]城市(斯坦纳树)

传送门
题意:给一个nnn个点mmm条边的带权无向图,让你构造一棵最小生成树使得kkk个关键点都在上面n≤1e5,m≤2e5,k≤5n\le1e5,m\le2e5,k\le5n1e5,m2e5,k5


思路:
套上斯坦纳树板子:
fi,jf_{i,j}fi,j表示iii为根子树特殊点状态为jjj的最优值,然后分情况转移即可。

  1. 枚举jjj子集
  2. 枚举不同的根

第二种式子写出来就是最短路的松弛操作因此可以上dijkstradijkstradijkstra
代码:

#include<bits/stdc++.h>
#define ri register int
#define fi first
#define se second
using namespace std;
const int rlen=1<<18|1;
inline char gc(){
    static char buf[rlen],*ib,*ob;
    (ib==ob)&&(ob=(ib=buf)+fread(buf,1,rlen,stdin));
    return ib==ob?-1:*ib++;
}
inline int read(){
    int ans=0;
    char ch=gc();
    while(!isdigit(ch))ch=gc();
    while(isdigit(ch))ans=((ans<<2)+ans<<1)+(ch^48),ch=gc();
    return ans;
}
const int N=1e5+5;
typedef long long ll;
typedef pair<ll,int> pli;
typedef pair<int,int> pii;
const ll inf=1e18;
ll f[N][32],ans=inf;
int n,m,t;
bool vis[N];
vector<pii>e[N];
priority_queue<pli>q;
inline void dijkstra(int sta){
    for(ri i=1;i<=n;++i)vis[i]=0;
    while(!q.empty()){
        int x=q.top().se;
        q.pop();
        if(vis[x])continue;
        vis[x]=1;
        for(ri i=0,v;i<e[x].size();++i){
            if(f[v=e[x][i].fi][sta]>f[x][sta]+e[x][i].se){
                f[v][sta]=f[x][sta]+e[x][i].se;
                if(!vis[v])q.push(pli(-f[v][sta],v));
            }
        }
    }
}
int main(){
    n=read(),t=read(),m=read();
    for(ri i=1,up=1<<t;i<=n;++i)for(ri j=0;j<up;++j)f[i][j]=inf;
    for(ri i=1;i<=t;++i)f[read()][1<<(i-1)]=0;
    for(ri i=1,a,b,c;i<=m;++i)a=read(),b=read(),c=read(),e[a].push_back(pii(b,c)),e[b].push_back(pii(a,c));
    for(ri sta=0,up=1<<t;sta<up;++sta){
        for(ri i=1;i<=n;++i){
            for(ri j=sta&(sta-1);j;j=sta&(j-1))f[i][sta]=min(f[i][sta],f[i][j]+f[i][sta^j]);
            if(f[i][sta]^inf)q.push(pli(-f[i][sta],i));
        }
        dijkstra(sta);
        if(sta==up-1)for(ri i=1;i<=n;++i)ans=min(ans,f[i][sta]);
    }
    cout<<ans;
    return 0;
}

转载于:https://www.cnblogs.com/ldxcaicai/p/11020459.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值