zoukankan      html  css  js  c++  java
  • 8.10-DayT3游走(wander)

    题目大意

    lue..

    题解

    先跑一遍tarjan缩点,在新图中加入两个强连通分量之间的边,则此图为一个有向无环图(DAG)。则最终答案为1点所在的强连通分量或包括1点的几个强连通分量的点数之和。

     如果为几个强连通分量则由于该图为DAG而题中要求为从1点出发又回到1点,

    故路径中一定包含一条反向边。又由于强连同分量中的点彼此强连同,故该反向边一定为两个强连同分量之间的边。

     故路径为一条边+一条经过一点所在强连通分量的路径。

    图中每个点代表一个强连通分量。其中1为包含1点的强通分量。

    其中路径为k-->........3-->1-->2-->4-->......-->n,而反向边为边k-->n

     因此,最终答案即为求如上一条包含点最多的路径。 

    考虑边k-->n,边k-->n一定为缩点后强连通分量之间的边。如果首先求出路径长度则枚举边k-->n即可。而路径长度一定为k-->1的包含点最多的路径长度与1-->n的包含点最多的路径的点的个数之和减1点所在的强连通分量包含的点的个数。

     故可以预处理出1点所在的强连通分量到其他强连通分量的路径中最多包含点的个数,再将所有强连通分量间的边反向,求1点所在的强连通分量到其他强连通分量的路径中最多包含点的个数,既求其他强连通分量到1点所在的强连通分量的路径中最多包含点的个数。

     最后枚举所有强连通分量之间的边k-->n,答案为 max(f1[n]+f2[k]-size[bel[1]])

     注意:当f1f20时不更新答案因为如果为0则代表1点所在的强连通分量

    无法到达n点或k点。

     Tarjan时间复杂度为O(n+m),两次DAG上求最长路的时间复杂度为O(m)

    总体时间复杂度O(n+m)

    (hhh一看就不是我自己写的题解...改不动了hhh)

    #include<stack>
    #include<queue>
    #include<stdio.h>
    #include<string.h>
    #include<algorithm>
    using namespace std;
    #define N 110000
    int n,m,ans;
    int t1,t2,tot,scc,cnt;
    int head[N],to[2*N],nex[2*N];
    int deep[N],low[N],bel[N],vis[N];
    int ins[N],inq[N],size[N];
    int f1[N],f2[N];
    stack<int>s;
    queue<int>que;
    void add(int x,int y)
    {
        tot++;
        nex[tot]=head[x];
        head[x]=tot;
        to[tot]=y;
    }
    int tot1;
    int head1[N],to1[2*N],nex1[2*N],from1[2*N];
    void add1(int x,int y)
    {
        tot1++;
        nex1[tot1]=head1[x];
        head1[x]=tot1;
        to1[tot1]=y;
        from1[tot1]=x;
    }
    int tot2;
    int head2[N],to2[2*N],nex2[2*N],from2[2*N];
    void add2(int x,int y)
    {
        tot2++;
        nex2[tot2]=head2[x];
        head2[x]=tot2;
        to2[tot2]=y;
        from2[tot2]=x;
    }
    void tarjan(int x)
    {
        deep[x]=low[x]=++cnt;
        ins[x]=1;
        vis[x]=1;
        s.push(x);
        for(int i=head[x];i;i=nex[i])
        {
            if(ins[to[i]])
            low[x]=min(low[x],deep[to[i]]);
            else if(!vis[to[i]])
            {    
                tarjan(to[i]);
                low[x]=min(low[x],low[to[i]]);
            }
        }
        if(deep[x]==low[x])
        {
            scc++;
            int tmp=s.top();
            s.pop();
            size[scc]++;
            bel[tmp]=scc;
            ins[tmp]=0;
            while(tmp!=x)
            {
                tmp=s.top();
                s.pop();
                size[scc]++;
                ins[tmp]=0;
                bel[tmp]=scc;
            }
        }
    }
    int main()
    {
        freopen("wander.in","r",stdin);
        freopen("wander.out","w",stdout);
        scanf("%d%d",&n,&m);
        for(int i=1;i<=m;i++)
        {
            scanf("%d%d",&t1,&t2);
            add(t1,t2);
        }
        for(int i=1;i<=n;i++)
        {
            if(!vis[i])
            {
                cnt=0;
                tarjan(i);
            }
        }
        for(int i=1;i<=n;i++)
        {
            for(int j=head[i];j;j=nex[j])
            if(bel[i]!=bel[to[j]])
            {
                add1(bel[i],bel[to[j]]);
                add2(bel[to[j]],bel[i]);
            }
        }
        que.push(bel[1]);
        f1[bel[1]]=size[bel[1]];
        inq[bel[1]]=1;
        while(!que.empty())
        {
            int tmp=que.front();
            que.pop();
            inq[tmp]=0;
            for(int i=head1[tmp];i;i=nex1[i])
            if(f1[to1[i]]<f1[tmp]+size[to1[i]])
            {
                f1[to1[i]]=f1[tmp]+size[to1[i]];        
                if(!inq[to1[i]])
                {
                    inq[to1[i]]=1;
                    que.push(to1[i]);
                }
            }
        }
        memset(inq,0,sizeof(inq));
        que.push(bel[1]);
        inq[bel[1]]=1;
        f2[bel[1]]=size[bel[1]];
        while(!que.empty())
        {
            int tmp=que.front();
            que.pop();
            inq[tmp]=0;
            for(int i=head2[tmp];i;i=nex2[i])
            if(f2[to2[i]]<f2[tmp]+size[to2[i]])
            {
                f2[to2[i]]=f2[tmp]+size[to2[i]];
                if(!inq[to2[i]])
                {
                    inq[to2[i]]=1;
                    que.push(to2[i]);
                }
            }
        }
        ans=max(ans,size[bel[1]]);
        for(int i=1;i<=tot2;i++)
        if(f2[to2[i]]&&f1[from2[i]])
        {
            if(f2[to2[i]]+f1[from2[i]]-size[bel[1]]>ans)
            ans=f2[to2[i]]+f1[from2[i]]-size[bel[1]];
        }
        printf("%d",ans);
        return 0;
    }
    std太强了
  • 相关阅读:
    HDU 2836 Traversal 简单DP + 树状数组
    UVa 1402 Runtime Error 伸展树
    UVa 11922
    HDU 4358 Boring counting 树状数组+思路
    HDU 4351 Digital root 线段树区间合并
    LA 6187
    UPC 2224 / “浪潮杯”山东省第四届ACM大学生程序设计竞赛 1008 Boring Counting 主席树
    max 宏定义取消:error C2589: error C2059: 语法错误 : “::”
    QT+VTK 对接使用
    标准C++中的string类的用法总结
  • 原文地址:https://www.cnblogs.com/darlingroot/p/11331887.html
Copyright © 2011-2022 走看看