题目大意
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]])
注意:当f1或f2为0时不更新答案因为如果为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;
}
