• 洛谷P3119 草鉴定


    这个题调了一天。。

    传送门

    读完题目之后我们不难想出这个题是个tarjan缩点问题,因为尽量多的经过草场,所以一号点所在的强连通分量里左右的点都是不需要在进行走逆向边,所能到达的。

    然后问题就落在怎么处理我们走这一次逆向边上。

    仔细看题目要求,题目要求我们必须从一号点出发,最后回到一号点。所以我想到了建一个逆向的图,虚拟出cnt + 1 , cnt+2,cnt+3.....n+cnt号点,建一个逆向图(用进行缩点之后的点集重新构造),因为我们求出进过草场最多是多少,所以我们将强连通分量中有多少个点在缩点中记录下来(num[i]:表示第ii个强连通分量中有多少个点),将它作为边权。

    最后我们只要从1号点所在的强连通分量开始,跑一边最长路就好了,最后输出从1号点到我们虚拟出来的1 + cnt号点就好了。

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    #include <queue>
    #include <algorithm>
    using namespace std;
    const int maxn = 2e5 + 5;
    
    inline int read(){
        char ch = getchar();
        int f = 1 , x = 0;
        while(ch > '9' || ch < '0'){if(ch == '-')f = -1;ch = getchar();}
        while(ch >= '0' && ch <= '9'){x = (x << 1) + (x << 3) + ch - '0';ch = getchar();}
        return x * f;
    }
    
    int n,m,x,y;
    int head[maxn],tot,head2[maxn],tot2;
    int ans;
    
    struct Edge{
        int from,to,next;
    }edge[maxn << 1],edge2[maxn << 1]; 
    
    void add(int u,int v){
        edge[++tot].from = u;
        edge[tot].to = v;
        edge[tot].next = head[u];
        head[u] = tot;
    }
    
    void add_edge(int u,int v){
        edge2[++tot2].from = u;
        edge2[tot2].to = v;
        edge2[tot2].next = head2[u];
        head2[u] = tot2;
    }
    
    int dfn[maxn],low[maxn],ind;
    int belong[maxn],num[maxn],cnt,stack[maxn],top;
    bool ins[maxn];
    
    void tarjan(int x){
        low[x] = dfn[x] = ++ind;
        ins[x] = true;
        stack[++top] = x;
        for(int i=head[x];i;i=edge[i].next){
            int v = edge[i].to;
            if(ins[v])  low[x] = min(low[x] , dfn[v]);
            if(!dfn[v]) {
                tarjan(v);
                low[x] = min(low[x] , low[v]);
            } 
        }
        int k = 0;
        if(dfn[x] == low[x]){
            cnt++;
            do{
                k = stack[top];
                num[cnt]++;
                top--;
                ins[k] = false;
                belong[k] = cnt;
            }  while(k != x);
        }
    }
    
    int dis[maxn];
    bool vis[maxn];
    
    void spfa(int s){
        queue<int> q;
        q.push(s);
        dis[s] = 0;
        vis[s] = true;
        while(!q.empty()){
            int cur = q.front();
            q.pop();  vis[cur] = false;
            for(int i=head2[cur];i;i=edge2[i].next){
                int v = edge2[i].to;
                if(dis[v] < dis[cur] + num[cur]){
                    dis[v] = dis[cur] + num[cur];
                    if(vis[v] == 0){
                        q.push(v);
                        vis[v] = true;
                    }
                }
            }
        }
    }
    
    int main(){
        n = read(); m = read();
        for(int i=1;i<=m;i++){
            x = read(); y =read();
            add(x , y);
        }
        for(int i=1;i<=n;i++)
           if(!dfn[i])  tarjan(i);
        for(int i=1;i<=n;i++)
           num[i + cnt] = num[i];
        for(int i=1;i<=m;i++)
            if(belong[edge[i].from] != belong[edge[i].to]){
                add_edge(belong[edge[i].from] , belong[edge[i].to]);
                add_edge(belong[edge[i].to] , belong[edge[i].from] + cnt);
                add_edge(belong[edge[i].from] + cnt , belong[edge[i].to] + cnt);
            }
        spfa(belong[1]);
        printf("%d
    ",dis[belong[1] + cnt]);
        return 0;
    }
    顺风不浪,逆风不怂。
  • 相关阅读:
    案例十:shell编写nginx服务启动程序
    Linux在实际中的应用
    案例九:shell脚本自动创建多个新用户,并设置密码
    数据架构的演变
    第一个Struts2程序
    关于eclipse导入Tomact报404的问题
    单选框 RadioButton
    EditText编辑框
    Button控件的三种点击事件
    1319: 同构词
  • 原文地址:https://www.cnblogs.com/Stephen-F/p/9864613.html
Copyright © 2020-2023  润新知