[Usaco2015 Jan]Grass Cownoisseur

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

数据范围:$1\le n, m\le 10^5$。


题解

先$tarjan$缩强连通分量,因为每一个$SCC$只要能到一个点就能到整个$SCC$。

接下来我们发现,我们操作的边的两个端点会满足如下性质:

这条有向边的起点可以到$1$号点所在$SCC$。

这条有向边的重点可以被$1$号点所在$SCC$到达。

故此,我们再缩完点之后,先对原图弄一遍拓扑序$DP$,求出$1$号点所在$SCC$到每个点的最长路。

再建反边重新跑拓扑序$DP$,求出每个点到$1$号点所在$SCC$的最长路。

暴力枚举边更新即可。

代码

#include <bits/stdc++.h>

#define N 100010 

using namespace std;

int dep[N], low[N], st[N], top, cnt, blg[N], sz[N], f1[N], f2[N], d1[N], d2[N];

int Number;

bool ins[N];

struct Node {
int x, y;
}e[N]; struct Edge {
int head[N], to[N << 1], nxt[N << 1], tot;
inline void add(int x, int y) {
to[ ++ tot] = y;
nxt[tot] = head[x];
head[x] = tot;
}
}G1,G2,G3; char *p1, *p2, buf[100000]; #define nc() (p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, 100000, stdin), p1 == p2) ? EOF : *p1 ++ ) int rd() {
int x = 0, f = 1;
char c = nc();
while (c < 48) {
if (c == '-')
f = -1;
c = nc();
}
while (c > 47) {
x = (((x << 2) + x) << 1) + (c ^ 48), c = nc();
}
return x * f;
} void tarjan(int p) {
st[ ++ top] = p;
ins[p] = true;
low[p] = dep[p] = ++cnt;
for (int i = G1.head[p]; i; i = G1.nxt[i]) {
if (!dep[G1.to[i]])
tarjan(G1.to[i]), low[p] = min(low[p], low[G1.to[i]]);
else if (ins[G1.to[i]])
low[p] = min(low[p], dep[G1.to[i]]);
}
if (dep[p] == low[p]) {
int t;
Number ++ ;
do {
t = st[top -- ];
ins[t] = false;
blg[t] = Number;
sz[Number] ++ ;
} while(t != p);
}
} queue<int> q; void dp1() {
while (!q.empty()) {
q.pop();
}
memset(f1, 0xef, sizeof f1);
for (int i = 1; i <= Number; i ++ ) {
if (!d1[i]) {
q.push(i);
}
}
f1[blg[1]] = 0;
while (!q.empty()) {
int x = q.front();
q.pop();
f1[x] += sz[x];
for (int i = G2.head[x]; i; i = G2.nxt[i]) {
f1[G2.to[i]] = max(f1[G2.to[i]], f1[x]);
d1[G2.to[i]] -- ;
if (!d1[G2.to[i]]) {
q.push(G2.to[i]);
}
}
}
} void dp2() {
while (!q.empty()) {
q.pop();
}
for (int i = 1; i <= Number; i ++ ) {
if (!d2[i]) {
q.push(i);
}
}
memset(f2, 0xef, sizeof f2);
f2[blg[1]] = 0;
while (!q.empty()) {
int x = q.front();
q.pop();
f2[x] += sz[x];
for (int i = G3.head[x]; i; i = G3.nxt[i]) {
f2[G3.to[i]] = max(f2[G3.to[i]], f2[x]);
d2[G3.to[i]] -- ;
if (!d2[G3.to[i]]) {
q.push(G3.to[i]);
}
}
}
} int main() {
int n = rd(), m = rd();
if (!n)
puts("1"), exit(0);
for (int i = 1; i <= m; i ++ ) {
e[i].x = rd(), e[i].y = rd();
G1.add(e[i].x, e[i].y);
}
for (int i = 1; i <= n; i ++ ) {
if (!dep[i]) {
tarjan(i);
}
}
// for (int i = 1; i <= n; i ++ ) {
// printf("%d ",blg[i]);
// }
// puts("");
for (int i = 1; i <= m; i ++ ) {
e[i].x = blg[e[i].x];
e[i].y = blg[e[i].y];
if (e[i].x != e[i].y) {
// printf("%d %d\n", e[i].x, e[i].y);
G2.add(e[i].x, e[i].y);
d1[e[i].y] ++ ;
G3.add(e[i].y, e[i].x);
d2[e[i].x] ++ ;
}
}
dp1();
dp2();
// for (int i = 1; i <= Number; i ++ ) {
// printf("%d %d\n", f1[i], f2[i]);
// }
int ans = sz[blg[1]];
for (int i = 1; i <= m; i ++ ) {
if (e[i].x != e[i].y) {
ans = max(ans, f1[e[i].y] + f2[e[i].x] - sz[blg[1]]);
}
}
cout << ans << endl ;
return 0;
}

小结:比较好想的一道题,需要注意的是两边拓扑序$dp$需要清队列。

[bzoj3887][Usaco2015 Jan]Grass Cownoisseur_trajan_拓扑排序_拓扑序dp的更多相关文章

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

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

  2. bzoj3887: [Usaco2015 Jan]Grass Cownoisseur

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

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

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

  4. BZOJ_3887_[Usaco2015 Jan]Grass Cownoisseur_强连通分量+拓扑排序+DP

    BZOJ_3887_[Usaco2015 Jan]Grass Cownoisseur_强连通分量+拓扑排序+DP Description In an effort to better manage t ...

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

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

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

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

  7. Codeforces 919D Substring ( 拓扑排序 && DAG上的DP )

    题意 : 给出含有 N 个点 M 条边的图(可能不连通或者包含环),每个点都标有一个小写字母编号,然后问你有没有一条路径使得路径上重复字母个数最多的次数是多少次,例如图上有条路径的顶点标号顺序是  a ...

  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. [Usaco2015 Jan]Grass Cownoisseur Tarjan缩点+SPFA

    考试的时候忘了缩点,人为dfs模拟缩点,没想到竟然跑了30分,RB爆发... 边是可以重复走的,所以在同一个强连通分量里,无论从那个点进入从哪个点出,所有的点一定能被一条路走到. 要使用缩点. 然后我 ...

随机推荐

  1. Luogu P1092 虫食算 爆搜

    心路历程:震惊,我竟然得了$90$分!!...康康数据...奥..(忽然有了邪恶的卡数据的想法) 于是把$for(int \space i=0;i<n;++i)$改成了$for(int \spa ...

  2. PHP mysqli_fetch_assoc() 函数

    从结果集中取得一行作为关联数组: <?php // 假定数据库用户名:root,密码:123456,数据库:RUNOOB $con=mysqli_connect("localhost& ...

  3. 浅谈C语言和C++中“类”的区别

    在C语言中,没有“类”的概念,但是可以由结构体struct构造出我们所需要的数据类型,struct可以组合不同的数据类型,可以看作是C语言中的“类”. 下面是C语言中的结构体的实例. #include ...

  4. 初学Javascript,写一个简易的登陆框

    <!--下面是源代码--> <!DOCTYPE html> <html> <head> <meta charset = "utf-8&q ...

  5. Web安全(白帽子讲)之第二篇

    第二章:浏览器安全 2.1.同源策略 是一种约定,它是浏览器最核心也是最基本的安全功能. web是构建在同源策略的基础之上,浏览器只是针对同源策略的一种实现 影响“源” 的因素有:host(域名或IP ...

  6. Zookeeper系列(十)zookeeper的服务端启动详述

    作者:leesf    掌控之中,才会成功:掌控之外,注定失败.出处:http://www.cnblogs.com/leesf456/p/6105276.html尊重原创,大家功能学习进步:  一.前 ...

  7. ARP输入 之 arp_rcv

    概述 arp_rcv是ARP包的入口函数,ARP模块在二层注册了类型为ETH_P_ARP的数据包回调函数arp_rcv,当收到ARP包时,二层进行分发,调用arp_rcv: arp_rcv对ARP输入 ...

  8. centos7 python2升级python3

    安装前的准备环境 yum install gcc patch libffi-devel python-devel zlib-devel bzip2-devel openssl-devel ncurse ...

  9. PHP ajax 实现三级联动

    在一个单独JS页面中,利用ajax实现三级联动,做成一个三级联动形式,以便于以后随时调用 JS代码: $(document).ready(function(e) { $("#sanji&qu ...

  10. javascript 数据分组

    一.静态数据 [ {"id":"1001","name":"值1","value":"11 ...