题目大意

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[*N],nex[*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[*N],nex1[*N],from1[*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[*N],nex2[*N],from2[*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]=;
vis[x]=;
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]=;
while(tmp!=x)
{
tmp=s.top();
s.pop();
size[scc]++;
ins[tmp]=;
bel[tmp]=scc;
}
}
}
int main()
{
freopen("wander.in","r",stdin);
freopen("wander.out","w",stdout);
scanf("%d%d",&n,&m);
for(int i=;i<=m;i++)
{
scanf("%d%d",&t1,&t2);
add(t1,t2);
}
for(int i=;i<=n;i++)
{
if(!vis[i])
{
cnt=;
tarjan(i);
}
}
for(int i=;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[]);
f1[bel[]]=size[bel[]];
inq[bel[]]=;
while(!que.empty())
{
int tmp=que.front();
que.pop();
inq[tmp]=;
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]]=;
que.push(to1[i]);
}
}
}
memset(inq,,sizeof(inq));
que.push(bel[]);
inq[bel[]]=;
f2[bel[]]=size[bel[]];
while(!que.empty())
{
int tmp=que.front();
que.pop();
inq[tmp]=;
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]]=;
que.push(to2[i]);
}
}
}
ans=max(ans,size[bel[]]);
for(int i=;i<=tot2;i++)
if(f2[to2[i]]&&f1[from2[i]])
{
if(f2[to2[i]]+f1[from2[i]]-size[bel[]]>ans)
ans=f2[to2[i]]+f1[from2[i]]-size[bel[]];
}
printf("%d",ans);
return ;
}

std太强了

8.10-DayT3游走(wander)的更多相关文章

  1. 【BZOJ-3143】游走 高斯消元 + 概率期望

    3143: [Hnoi2013]游走 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2264  Solved: 987[Submit][Status] ...

  2. 3143: [Hnoi2013]游走 - BZOJ

    Description 一个无向连通图,顶点从1编号到N,边从1编号到M. 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一个顶点, ...

  3. bzoj 3143: [Hnoi2013]游走 高斯消元

    3143: [Hnoi2013]游走 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1026  Solved: 448[Submit][Status] ...

  4. 数学(概率):HNOI2013 游走

    [题目描述] 一个无向连通图,顶点从1编号到N,边从1编号到M. 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一个顶点,获得等于这 ...

  5. 介绍一个全局最优化的方法:随机游走算法(Random Walk)

    1. 关于全局最优化求解   全局最优化是一个非常复杂的问题,目前还没有一个通用的办法可以对任意复杂函数求解全局最优值.上一篇文章讲解了一个求解局部极小值的方法--梯度下降法.这种方法对于求解精度不高 ...

  6. [补档][Hnoi2013]游走

    [Hnoi2013]游走 题目 一个无向连通图,顶点从1编号到N,边从1编号到M. 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一 ...

  7. 游走[HNOI2013]

    [题目描述] 一个无向连通图,顶点从1编号到N,边从1编号到M. 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一个顶点,获得等于这 ...

  8. bzoj 3143: [Hnoi2013]游走

    Description 一个无向连通图,顶点从1编号到N,边从1编号到M. 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一个顶点, ...

  9. BZOJ 3143: [Hnoi2013]游走 [概率DP 高斯消元]

    一个无向连通图,顶点从1编号到N,边从1编号到M. 小Z在该图上进行随机游走,初始时小Z在1号顶点,每一步小Z以相等的概率随机选 择当前顶点的某条边,沿着这条边走到下一个顶点,获得等于这条边的编号的分 ...

随机推荐

  1. 2018ICPC南京站Problem G. Pyramid

    题意: 找有多少个等边三角形 解析: 首先打标找规律,然后对式子求差分 0,1,5,15,35,70,126,210... 1,4,10,20,35,56... 3,6,10,15,21... 3,4 ...

  2. centos7搭建zabbix

    参考:https://blog.csdn.net/xiaocong66666/article/details/82818893 安装所需的依赖包即可: yum install gcc gcc-c++ ...

  3. TCL namespace

    命名空间可从Tcl 8.0版开始使用.引入命名空间之前,有一个全局范围.现在有了命名空间,我们可以分区全局范围. 创建命名空间: 结果:33 嵌套命名空间: 结果: test1 test2 导入命名空 ...

  4. Linq To Sqlite使用心得

    若要使用Linq To Sqlite类库,可以安装Devart Linq Connect Model,如图: 新建这个Model就可以和Linq To Sql一样使用Linq模型,下载地址:https ...

  5. 在SQL2005中修改数据库名称

    SQL Server 2005中有个数据库HT_WisdomDataBaseCenter,现在要将其改名为HT_NBSBOneNetcs1步骤:(1) 分离数据库:打开management studi ...

  6. ReLU(inplace=True),这里的inplace=true的意思

    ReLU(inplace=True),这里的inplace=true的意思 待办 inplace=True means that it will modify the input directly, ...

  7. Python之路Day06

    小数据池 == 判断两个值是否相等 is -- 是,判断两边的内存地址是否相同 a=10 b=10 print(a is b) id() -- 查看内存地址 代码块 一个py文件,一个函数,一个模块, ...

  8. Visual Studio Code快捷键大全

    原文链接:https://segmentfault.com/a/1190000007688656 常用 General 按 Press 功能 Function Ctrl + Shift + P,F1 ...

  9. MySQL认知

    MySQL 认识MySQL MySQL是什么? MySQL是最流行的关系型数据库管理系统,在WEB应用方面MySQL是最好的RDBMS(Relational Database Management S ...

  10. OPC通信原理在数采中的应用

    OPC通信原理在数采中的应用 OPC是Object Linking and Embedding(OLE)for Process Control的缩写,它是微软公司的对象链接和嵌入技术在过程控制方面的应 ...