\(\mathcal{Description}\)

  Link.

  给定 \(n\) 个点 \(m\) 条边的有向弱连通图。称一个点是“好点”当且仅当从该点出发,不存在到同一点的两条不同简单路径。求出所有好点,但若好点个数少于 \(n \times 20\%\),仅输出 -1

  多测,\(n,\sum_{}^{} n \le10^5\),\(m,\sum_{}^{} m\le2\times10^5\)。

\(\mathcal{Solution}\)

  先来想一想如何判断某个点 \(r\) 是好点。以 \(r\) 为根建出任意一棵外向生成树,不难发现 \(r\) 是好点,当且仅当树外不存在横叉边(有向 DFS 树是可能存在横叉边的)。

  假设我们已经得到了一点 \(r\) 为好点,如何判断出其他点是否是好点呢?考虑在刚才那棵外向生成树上的某一非根结点 \(u\),若从 \(u\) 子树内向 \(u\) 的祖先的返祖边多于一条,就显然不合法,否则若 \(u\) 能到达最高的祖先合法,\(u\) 也必然合法。

  最后一个问题,怎么找到一个 \(r\) 呢?考虑到 \(20\%\) 的限制,随机 \(100\) 个点进行好点测验,如果都不是好点直接输出 -1。错误率为 \(\left( \frac{4}{5} \right)^{100}\),非常可观。

  复杂度 \(\mathcal O(100\sum_{}^{} n)\)。

\(\mathcal{Code}\)

/* Clearink */

#include <cstdio>
#include <random> inline int rint () {
int x = 0, f = 1; char s = getchar ();
for ( ; s < '0' || '9' < s; s = getchar () ) f = s == '-' ? -f : f;
for ( ; '0' <= s && s <= '9'; s = getchar () ) x = x * 10 + ( s ^ '0' );
return x * f;
} template<typename Tp>
inline void wint ( Tp x ) {
if ( x < 0 ) putchar ( '-' ), x = ~ x + 1;
if ( 9 < x ) wint ( x / 10 );
putchar ( x % 10 ^ '0' );
} const int MAXN = 1e5, MAXM = 2e5;
int n, m, ecnt, head[MAXN + 5], vtag[MAXN + 5], upc[MAXN + 5], top[MAXN + 5];
bool bad[MAXN + 5];
std::mt19937 rnd ( 20050913 ); struct Edge { int to, nxt; } graph[MAXM + 5]; inline void link ( const int s, const int t ) {
graph[++ ecnt] = { t, head[s] };
head[s] = ecnt;
} inline bool check ( const int u ) {
vtag[u] = 1;
for ( int i = head[u], v; i; i = graph[i].nxt ) {
if ( !vtag[v = graph[i].to] ) {
if ( !check ( v ) ) return false;
} else if ( vtag[v] == 2 ) {
return false;
}
}
vtag[u] = 2;
return true;
} inline void mark ( const int u ) {
top[u] = u;
for ( int i = head[u], v; i; i = graph[i].nxt ) {
if ( !vtag[v = graph[i].to] ) {
vtag[v] = vtag[u] + 1;
mark ( v ), upc[u] += upc[v];
if ( vtag[top[v]] < vtag[top[u]] ) top[u] = top[v];
} else {
++ upc[u], -- upc[v];
if ( vtag[v] < vtag[top[u]] ) top[u] = v;
}
}
} inline int spread ( const int u ) {
vtag[u] = 1, bad[u] = upc[u] > 1;
int ret = !( bad[u] |= bad[top[u]] );
for ( int i = head[u], v; i; i = graph[i].nxt ) {
if ( !vtag[v = graph[i].to] ) {
ret += spread ( v );
}
}
return ret;
} inline void clear () {
ecnt = 0;
for ( int i = 1; i <= n; ++ i ) {
head[i] = vtag[i] = upc[i] = top[i] = bad[i] = 0;
}
} int main () {
for ( int T = rint (); T --; ) {
clear ();
n = rint (), m = rint ();
for ( int i = 1, u, v; i <= m; ++ i ) {
u = rint (), v = rint ();
link ( u, v );
}
int rt = 0;
for ( int i = 1; i <= 100 && !rt; ++ i ) {
int u = rnd () % n + 1;
for ( int i = 1; i <= n; ++ i ) vtag[i] = 0;
if ( check ( u ) ) rt = u;
}
if ( !rt ) { puts ( "-1" ); continue; }
for ( int i = 1; i <= n; ++ i ) vtag[i] = 0;
vtag[rt] = 1, mark ( rt );
for ( int i = 1; i <= n; ++ i ) vtag[i] = 0;
int cnt = spread ( rt );
if ( cnt * 5 < n ) { puts ( "-1" ); continue; }
for ( int i = 1, f = 0; i <= n; ++ i ) {
if ( !bad[i] ) {
if ( f ) putchar ( ' ' );
f = 1, printf ( "%d", i );
}
}
putchar ( '\n' );
}
return 0;
}

\(\mathcal{Details}\)

  这种随机乱搞题得放开点脑洞啊,看到这种 \(20\%\) 之类的奇怪限制就可以往这方面想啦。

Solution -「CF 1361E」James and the Chase的更多相关文章

  1. Solution -「CF 1342E」Placing Rooks

    \(\mathcal{Description}\)   Link.   在一个 \(n\times n\) 的国际象棋棋盘上摆 \(n\) 个车,求满足: 所有格子都可以被攻击到. 恰好存在 \(k\ ...

  2. Solution -「CF 1622F」Quadratic Set

    \(\mathscr{Description}\)   Link.   求 \(S\subseteq\{1,2,\dots,n\}\),使得 \(\prod_{i\in S}i\) 是完全平方数,并最 ...

  3. Solution -「CF 923F」Public Service

    \(\mathscr{Description}\)   Link.   给定两棵含 \(n\) 个结点的树 \(T_1=(V_1,E_1),T_2=(V_2,E_2)\),求一个双射 \(\varph ...

  4. Solution -「CF 923E」Perpetual Subtraction

    \(\mathcal{Description}\)   Link.   有一个整数 \(x\in[0,n]\),初始时以 \(p_i\) 的概率取值 \(i\).进行 \(m\) 轮变换,每次均匀随机 ...

  5. Solution -「CF 1586F」Defender of Childhood Dreams

    \(\mathcal{Description}\)   Link.   定义有向图 \(G=(V,E)\),\(|V|=n\),\(\lang u,v\rang \in E \Leftrightarr ...

  6. Solution -「CF 1237E」Balanced Binary Search Trees

    \(\mathcal{Description}\)   Link.   定义棵点权为 \(1\sim n\) 的二叉搜索树 \(T\) 是 好树,当且仅当: 除去最深的所有叶子后,\(T\) 是满的: ...

  7. Solution -「CF 623E」Transforming Sequence

    题目 题意简述   link.   有一个 \(n\) 个元素的集合,你需要进行 \(m\) 次操作.每次操作选择集合的一个非空子集,要求该集合不是已选集合的并的子集.求操作的方案数,对 \(10^9 ...

  8. Solution -「CF 1023F」Mobile Phone Network

    \(\mathcal{Description}\)   Link.   有一个 \(n\) 个结点的图,并给定 \(m_1\) 条无向带权黑边,\(m_2\) 条无向无权白边.你需要为每条白边指定边权 ...

  9. Solution -「CF 599E」Sandy and Nuts

    \(\mathcal{Description}\)   Link.   指定一棵大小为 \(n\),以 \(1\) 为根的有根树的 \(m\) 对邻接关系与 \(q\) 组 \(\text{LCA}\ ...

随机推荐

  1. spring cloud bus 消息总线 动态刷新配置文件 【actuator 与 RabbitMQ配合完成】

    1.前言 单机刷新配置文件,使用actuator就足够了 ,但是 分布式微服务 不可能是单机 ,将会有很多很多的工程 ,无法手动一个一个的发送刷新请求, 因此引入了消息中间件 ,常用的 消息中间件 是 ...

  2. Linux上天之路(十二)之服务管理

    主要内容 服务介绍 独立服务 非独立服务 1. 服务介绍 服务:常驻在内存中的程序,且可以提供一些系统或网络功能,那就是服务. 计算机中的系统服务有很多,比如: apache提供web服务 ftp提供 ...

  3. 接口神器之 Json Server 详细指南

    简介 json-server 是一款小巧的接口模拟工具,一分钟内就能搭建一套 Restful 风格的 api,尤其适合前端接口测试使用. 只需指定一个 json 文件作为 api 的数据源即可,使用起 ...

  4. php中使用CURL之php curl详解

    curl是个什么东西?简单地说就是,curl是一个库,能让你通过URL和许多不同种的服务器进行勾搭.搭讪和深入交流,并且还支持许多协议.并且人家还说了curl可以支持https认证.http post ...

  5. 1012day-人口普查系统

    1.name.jsp <%@ page language="java" contentType="text/html; charset=UTF-8" pa ...

  6. thanos的日志能不能打到文件里面去?

    不行. thanos/pkg/logging/logger.go: logger = log.NewLogfmtLogger(log.NewSyncWriter(os.Stderr)) if logF ...

  7. 切换不同的echarts时,出现图标缩小,报警告,Can’t get dom width or height!

    出现这样的原因是因为,在切换的时候,图表所对应的标签还没有显示出来,最好将代码放在$nextick里面执行,并且,采用使用v-if进行切换 转载:https://www.pianshen.com/ar ...

  8. Python解释器下载安装教程

    简介: 自从20世纪90年代初Python语言诞生至今,它已被逐渐广泛应用于系统管理任务的处理和Web编程.2021年10月,语言流行指数的编译器Tiobe将Python加冕为最受欢迎的编程语言,20 ...

  9. linux关闭防火墙与开启防火墙命令

    一:下面是red hat/CentOs7关闭防火墙的命令! 1:查看防火状态 systemctl status firewalld service iptables status 2:暂时关闭防火墙 ...

  10. linux正则转换csv文件