[JZOJ3400] 【GDOI2014模拟】旅行

题目

题目大意

给你一个图,让你选择权值和最小的边,使得\(1\)\(n\)\(2\)\(n-1\),……,\(K\)\(n-K+1\)联通。
\(K\leq 4\)


思考历程

一看到这题就觉得特别神仙……
然后去思考网络流……
搞出了一个最小割,后来发现这是错的……
匆匆打了个表,获得了这题的十分之一的分数。


正解

其实这题有水法,许多人是全排列+\(SPFA\),跑了一遍之后将路过的边清\(0\),继续跑。这样贪心显然是错的,反例也有,但是极其水的数据居然给了他们\(100\)分!
正解是DP。
题解中有个叫做\(stenir \ tree\)的东西,感觉似乎很强大。当然我不懂,我只会DP。
现在我们是要求出一个最小生成森林,使得一些点对联通。
考虑一棵树。显然,树的叶子节点一定是要求联通的节点(反过来倒不一定)。
于是我们就试着DP……
\(f_{S,i}\)表示当前这棵树的根节点为\(i\),并且\(S\)集合中的所有点连在了一起的最小代价
转移的时候就是两棵树的根节点连在一起,也就是\(f_{S',i}+w(i,j)+f_{S-S',j}\to f_{S,i}\)
\(S'\)\(S\)的子集。(枚举\(S\)\(S'\)的时间复杂度是\(3^k\)的,\(k\)表示位数,具体实现比较巧妙,见代码)
然后这道题就差不多完了。注意,转移的时候一般是\(S\)从低到高转移,但也会向\(S\)相等的状态转移,这就意味着会有后效性。所以,对于同层的转移,我们特殊地用最短路算法来处理。


代码

using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <climits>
inline void update(int &a,int b){a>b?a=b:0;}
#define N 10010
#define M 10010
int n,m,K;
struct EDGE{
    int to,len;
    EDGE *las;
} e[M*2];
int ne;
EDGE *last[N];
inline void link(int u,int v,int len){
    e[ne]={v,len,last[u]};
    last[u]=e+ne++;
}
int f[256][N],*dis;
struct Node{
    int x,dis;
} h[1000001];
int nh;
inline bool cmph(const Node &son,const Node &fa){
    return son.dis>fa.dis;
}
int mn[256],ans[256];
inline bool ok(int s){
    for (int i=0;i<K;++i)
        if ((s>>i&1)^(s>>i+K&1))
            return 0;
    return 1;
}
int main(){
    freopen("in.txt","r",stdin); 
    scanf("%d%d%d",&n,&m,&K);
    for (int i=1;i<=m;++i){
        int u,v,len;
        scanf("%d%d%d",&u,&v,&len);
        link(u,v,len),link(v,u,len);
    }
    memset(f,63,sizeof f);
    for (int i=1;i<=K;++i)
        f[1<<i-1][i]=f[1<<K+i-1][n-i+1]=0;
    for (int i=K+1;i<=n-K;++i)
        f[0][i]=0;
    for (int s=1;s<1<<K*2;++s){
        for (int i=1;i<=n;++i)
            for (int s1=(s-1)&s;s1;s1=(s1-1)&s) 
                if (f[s1][i]<0x3f3f3f3f)
                    for (EDGE *ei=last[i];ei;ei=ei->las)
                        update(f[s][i],f[s1][i]+ei->len+f[s-s1][ei->to]);
        dis=f[s];
        nh=0;
        for (int i=1;i<=n;++i)
            h[nh++]={i,dis[i]};
        while (nh){
            int x=h[0].x,disx=h[0].dis;
            pop_heap(h,h+nh--,cmph);
            if (disx>dis[x])
                continue;
            for (EDGE *ei=last[x];ei;ei=ei->las)
                if (disx+ei->len<dis[ei->to]){
                    dis[ei->to]=disx+ei->len;
                    h[nh++]={ei->to,dis[ei->to]};
                    push_heap(h,h+nh,cmph);
                }
        }
        mn[s]=INT_MAX;
        for (int i=1;i<=n;++i)
            update(mn[s],dis[i]);
    }
    memset(ans,63,sizeof ans);
    ans[0]=0;
    for (int s=1;s<1<<K*2;++s)
        for (int s1=s;s1;s1=(s1-1)&s)
            if (ok(s-s1) && ok(s1))
                update(ans[s],ans[s-s1]+mn[s1]);
    if (ans[(1<<K*2)-1]<0x3f3f3f3f)
        printf("%d\n",ans[(1<<K*2)-1]);
    else
        printf("-1\n");
    return 0;
}

总结

不是什么题都能用网络流做的……
DP也能玩出各种花样……

转载于:https://www.cnblogs.com/jz-597/p/11183933.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值