考试的时候忘了缩点,人为dfs模拟缩点,没想到竟然跑了30分,RB爆发。。。

边是可以重复走的,所以在同一个强连通分量里,无论从那个点进入从哪个点出,所有的点一定能被一条路走到。 要使用缩点。

然后我们枚举每一条边,考虑如果将这条边反置的话,就是这条边的终点到1的点的权值(正向的)加上起点到1的点的权值(反向的);例:→→←→→

每个点到1的正向反向距离可以用两遍SPFA解决出来。

先使用tarjan缩点,记录每个点的权值,缩点后权值变为强连通分量中点的个数。缩完点之后重新建图,正向边存1,反向边存-1;

注意:在枚举每一条边求MAX时一定要判断它的起点和终点能否到达1。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<queue>
using namespace std;
#define pos(i,a,b) for(int i=(a);i<=(b);i++)
#define pos2(i,a,b) for(int i=(a);i>=(b);i--)
#define N 501000
int n,m;
struct haha
{
       int next,to,w;
};
haha edgechu[N],edge[N];
int head[N],headchu[N],cnt=1,cntchu=1,cntt,hea;
int belong[N],low[N],dfn[N],ji=1,stack[N],instack[N];
int point[N];
void add(int u,int v,int w)
{
     edge[cnt].to=v;
     edge[cnt].next=head[u];
     edge[cnt].w=w;
     head[u]=cnt++;
}
void addchu(int u,int v,int w)
{
     edgechu[cntchu].to=v;
     edgechu[cntchu].next=headchu[u];
     edgechu[cntchu].w=w;
     headchu[u]=cntchu++;
}
void tarjan(int now)
{
     low[now]=dfn[now]=ji;
     ji++;
     stack[++hea]=now;
     instack[now]=1;
     for(int v=headchu[now];v;v=edgechu[v].next)
     {
             int i=edgechu[v].to;
             if(dfn[i]==-1)
             {
                tarjan(i);
                low[now]=min(low[now],low[i]);
             }
             else
               if(instack[i])
                  low[now]=min(low[now],dfn[i]);
     }
     if(low[now]==dfn[now])
     {
         cntt++;
         int temp;
         while(1)
         {
           temp=stack[hea--];
           belong[temp]=cntt;
           point[cntt]++;
           instack[temp]=0;
           //cout<<"temp="<<temp<<"  cntt="<<cntt<<endl;
           if(temp==now)
             break;
         }
     }
}
int diszheng[N],disfan[N];
int flag[N];
void spfazheng(int x)
{
     queue<int> q;
     pos(i,1,n)
       diszheng[i]=0;
     memset(flag,0,sizeof(flag));
     diszheng[x]=point[x];
     q.push(x);
     flag[x]=1;
     int k;
     while(!q.empty())
     {
        k=q.front();
        for(int v=head[k];v;v=edge[v].next)
        {
          int i=edge[v].to;
          if(edge[v].w>0&&diszheng[i]<diszheng[k]+point[i])
          {
              diszheng[i]=diszheng[k]+point[i];
              if(!flag[i])
              {
                 q.push(i);
                 flag[i]=1;
              }
          }
        }
        flag[q.front()]=0;
        q.pop();
     }
}
void spfafan(int x)
{
     queue<int> q;
     pos(i,1,n)
       disfan[i]=0;
     memset(flag,0,sizeof(flag));
     disfan[x]=point[x];
     q.push(x);
     flag[x]=1;
     int k;
     while(!q.empty())
     {
        k=q.front();
        for(int v=head[k];v;v=edge[v].next)
        {
          int i=edge[v].to;
          if(edge[v].w<0&&disfan[i]<disfan[k]+point[i])
          {
              disfan[i]=disfan[k]+point[i];
              if(!flag[i])
              {
                 q.push(i);
                 flag[i]=1;
              }
          }
        }
        flag[q.front()]=0;
        q.pop();
     }
}
struct qian
{
   int from,to;
}cun[N];
int road;
int ans;
int main()
{
    //freopen("cown.in","r",stdin);
    //freopen("cown.out","w",stdout);
    scanf("%d%d",&n,&m);
    memset(dfn,-1,sizeof(dfn));
    pos(i,1,m)
    {
       int x,y;
       scanf("%d%d",&x,&y);
       addchu(x,y,1);
    }
    pos(i,1,n)
      if(dfn[i]==-1)
       tarjan(i);
    pos(i,1,n)
       for(int v=headchu[i];v;v=edgechu[v].next)
       {
          int j=edgechu[v].to;
          if(belong[i]!=belong[j])
          {
             add(belong[i],belong[j],1);
             add(belong[j],belong[i],-1);
             cun[++road].from=belong[i];
             cun[road].to=belong[j];
          }
       }
    spfafan(belong[1]);spfazheng(belong[1]);
    /*pos(i,1,cntt)
      cout<<"diszheng[i]="<<diszheng[i]<<"   disfan[i]="<<disfan[i]<<endl;*/
    pos(i,1,road)
    {
      if(diszheng[cun[i].to]>0&&disfan[cun[i].from]>0)
       ans=max(ans,diszheng[cun[i].to]+disfan[cun[i].from]-point[belong[1]]);
       //cout<<"ans="<<ans<<endl;
    }
    printf("%d",ans);
    //while(1);
    return 0;
}

  

[Usaco2015 Jan]Grass Cownoisseur Tarjan缩点+SPFA的更多相关文章

  1. BZOJ 3887: [Usaco2015 Jan]Grass Cownoisseur tarjan + spfa

    Code: #include <bits/stdc++.h> #define setIO(s) freopen(s".in","r",stdin) ...

  2. BZOJ3887 [Usaco2015 Jan] Grass Cownoisseur 【tarjan】【DP】*

    BZOJ3887 [Usaco2015 Jan] Grass Cownoisseur Description In an effort to better manage the grazing pat ...

  3. [补档][Usaco2015 Jan]Grass Cownoisseur

    [Usaco2015 Jan]Grass Cownoisseur 题目 给一个有向图,然后选一条路径起点终点都为1的路径出来,有一次机会可以沿某条边逆方向走,问最多有多少个点可以被经过? (一个点在路 ...

  4. bzoj3887: [Usaco2015 Jan]Grass Cownoisseur

    题意: 给一个有向图,然后选一条路径起点终点都为1的路径出来,有一次机会可以沿某条边逆方向走,问最多有多少个点可以被经过?(一个点在路径中无论出现多少正整数次对答案的贡献均为1) =>有向图我们 ...

  5. BZOJ 3887/Luogu P3119: [Usaco2015 Jan]Grass Cownoisseur (强连通分量+最长路)

    分层建图,反向边建在两层之间,两层内部分别建正向边,tarjan缩点后,拓扑排序求一次1所在强连通分量和1+n所在强联通分量的最长路(长度定义为路径上的强联通分量内部点数和).然后由于1所在强连通分量 ...

  6. [Usaco2015 Jan]Grass Cownoisseur 图论 tarjan spfa

    先缩点,对于缩点后的DAG,正反跑spfa,枚举每条边进行翻转即可 #include<cstdio> #include<cstring> #include<iostrea ...

  7. BZOJ3887 [Usaco2015 Jan]Grass Cownoisseur[缩点]

    首先看得出缩点的套路.跑出DAG之后,考虑怎么用逆行条件.首先可以不用,这样只能待原地不动.用的话,考虑在DAG上向后走,必须得逆行到1号点缩点后所在点的前面,才能再走回去. 于是统计从1号点缩点所在 ...

  8. 洛谷—— P3119 [USACO15JAN]草鉴定Grass Cownoisseur || BZOJ——T 3887: [Usaco2015 Jan]Grass Cownoisseur

    http://www.lydsy.com/JudgeOnline/problem.php?id=3887|| https://www.luogu.org/problem/show?pid=3119 D ...

  9. [bzoj3887][Usaco2015 Jan]Grass Cownoisseur_trajan_拓扑排序_拓扑序dp

    [Usaco2015 Jan]Grass Cownoisseur 题目大意:给一个有向图,然后选一条路径起点终点都为1的路径出来,有一次机会可以沿某条边逆方向走,问最多有多少个点可以被经过?(一个点在 ...

随机推荐

  1. Vue和Bootstrap的整合之路

    我是一个刚刚接触前端开发的新手,所以有必要记录如何将Bootstrap和Vue进行整合. 如果你是老手,请直接绕道而过.作为一个新手,里面的步骤,过程或者专业术语未必正确,如果你发现哪里错误了,请发邮 ...

  2. solr学习-基础环境搭建(一)

    目前网上关于solr6.+的安装教程很少,有些6.0之前的教程在应用到6.+的版本中出现很多的问题,所以特别整理出来这一片文章,希望能给各位码农一些帮助! 很少写些文章,如有不对的地方,还希望多多指导 ...

  3. [leetcode-628-Maximum Product of Three Numbers]

    Given an integer array, find three numbers whose product is maximum and output the maximum product. ...

  4. Example010实现浏览器兼容改内容的函数,自写

    <!-- 实例010实现浏览器兼容改内容的函数 --> <!DOCTYPE html> <html lang="en"> <head> ...

  5. JS综合练习

    练习一.任意数求和(最多输入十位数),输入999终止 运行代码 <!DOCTYPE html><html> <head> <meta charset=&quo ...

  6. 【summary】JQuery 相关css、ajax、数据操作函数或方法

    总结一下JQuery常用的函数方法,更加系统的整理一下. JQuery遍历的一些函数: 函数 描述 .add() 将元素添加到匹配元素的集合中. .andSelf() 把堆栈中之前的元素集添加到当前集 ...

  7. jap页面获取struts2中action中变量的值

    在jsp页面中可以通过ONGL表达式获取struts2中action处理后的变量的值,这是因为每一个action在初始化后都会放到strackcontext中,可以通过ONGL表达式取到值. 注意要在 ...

  8. java 关于 hashmap 的实现原理的测试

    网上关于HashMap的工作原理的文章多了去了,所以我也不打算再重复别人的文章.我就是有点好奇,我怎么样能更好的理解他的原理,或者说使用他的特性呢?最好的开发就是测试~ 虽说不详讲hashmap的工作 ...

  9. VB6之写注册表

    难免会遇到写注册表的情况,写了个实用点的RegWrite函数.为了减少代码量,用WScript.Shell取代了API来实现. 使用方式就在注释中了,就不再过多解释了.PS:注释比实现代码要丰富多了, ...

  10. C# Web.config配置

    使用 <!--M002 バッチを起動のPath配置--> <add key="BM0002_START_PATH" value="D:\BM0002\B ...