【51Nod 1815】【算法马拉松 23】调查任务
http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1815
tarjan缩点后在DAG上递推即可。
每个点维护所有根到它的路径上的值的最大值,严格次大值,最大的“根到这个点的一条路径中的严格次大值”(也就是答案)。
注意所有根到它的路径上的值的严格次大值不是答案。
时间复杂度\(O(n)\)。
#include<cstdio>
#include<bitset>
#include<cstring>
#include<algorithm>
using namespace std;
const int N = 400003;
const int M = 2000003;
struct node {int nxt, to;} E[M], E2[M], E3[M];
int cnt = 0, cnt3 = 0, point[N], point2[N], cur[N], du[N], point3[N];
void ins(int u, int v) {E[++cnt] = (node) {point[u], v}; point[u] = cnt;}
void ins2(int u, int v) {
E2[++cnt] = (node) {point2[u], v}; point2[u] = cnt;
E3[++cnt3] = (node) {point3[v], u}; point3[v] = cnt3;
}
bitset <N> inst;
int dfn[N], low[N], tot = 0, a[N], sta[N], statop, st[N], top, bel[N], n, m, q, s, fa[N];
void tarjan(int x) {
st[top = 1] = x; statop = 0;
while (top) {
int u = st[top];
if (!dfn[u]) {
dfn[u] = low[u] = ++cnt;
sta[++statop] = u;
inst[u] = 1;
}
if (cur[u]) {
int v = E[cur[u]].to;
if (!dfn[v]) fa[st[++top] = v] = u;
else if (inst[v]) low[u] = min(low[u], dfn[v]);
cur[u] = E[cur[u]].nxt;
} else {
low[fa[u]] = min(low[fa[u]], low[u]);
if (dfn[u] == low[u]) {
++tot;
while (sta[statop] != u) {
inst[sta[statop]] = 0;
bel[sta[statop]] = tot;
--statop;
}
inst[u] = 0;
bel[u] = tot;
--statop;
}
--top;
}
}
}
bitset <N> vis;
int max1[N], max2[N], max3[N], qu[N], A[4];
void BFS() {
int p = 0, q = 1;
vis[qu[1] = bel[s]] = 1;
while (p != q) {
int u = qu[++p];
for (int i = point2[u]; i; i = E2[i].nxt) {
int v = E2[i].to;
++du[v];
if (!vis[v]) {
vis[v] = 1;
qu[++q] = v;
}
}
}
p = 0; q = 1; qu[1] = bel[s];
while (p != q) {
int u = qu[++p];
for (int i = point3[u]; i; i = E3[i].nxt) {
int v = E3[i].to;
if (vis[v]) {
if (max1[v] > max1[u]) {max3[u] = max1[u]; max1[u] = max1[v];}
else if (max1[v] < max1[u] && max1[v] > max3[u]) max3[u] = max1[v];
if (max3[v] > max1[u]) {max3[u] = max1[u]; max1[u] = max3[v];}
else if (max3[v] < max1[u] && max3[v] > max3[u]) max3[u] = max3[v];
}
}
for (int i = point2[u]; i; i = E2[i].nxt) {
int v = E2[i].to;
max2[v] = max(max2[v], max2[u]);
if (max1[u] != max1[v]) max2[v] = max(max2[v], min(max1[u], max1[v]));
else max2[v] = max(max2[v], max(max3[u], max3[v]));
if (--du[v] == 0) qu[++q] = v;
}
}
}
int main() {
scanf("%d%d%d%d", &n, &m, &q, &s);
for (int i = 1; i <= n; ++i)
scanf("%d", a + i);
int u, v;
for (int i = 1; i <= m; ++i) {
scanf("%d%d", &u, &v);
ins(u, v);
}
cnt = 0;
for (int i = 1; i <= n; ++i)
cur[i] = point[i], max1[i] = max2[i] = -1;
for (int i = 1; i <= n; ++i)
if (!dfn[i]) tarjan(i);
cnt = 0;
for (int i = 1; i <= n; ++i)
for (int j = point[i]; j; j = E[j].nxt)
if (bel[i] != bel[E[j].to])
ins2(bel[i], bel[E[j].to]);
for (int i = 1; i <= n; ++i) {
int t = bel[i];
if (a[i] > max1[t]) max2[t] = max1[t], max1[t] = a[i];
else if (a[i] < max1[t] && a[i] > max2[t]) max2[t] = a[i];
}
for (int i = 1; i <= tot; ++i) max3[i] = max2[i];
BFS();
for (int i = 1; i <= q; ++i) {
scanf("%d", &u);
if (!vis[bel[u]]) printf("-1 ");
else if (max2[bel[u]] == -1) printf("0 ");
else printf("%d ", max2[bel[u]]);
}
return 0;
}
【51Nod 1815】【算法马拉松 23】调查任务的更多相关文章
- 51Nod 算法马拉松23 开黑记
惨啊……虽然开了半天黑,但是还是被dalao们踩了…… 第二次开黑,还是被卡在rank20了,我好菜啊……= = 写一写比赛经过吧…… 看到题之后习惯性都打开,A~D看上去似乎并没有什么思路,F应该是 ...
- 【51Nod 1756】【算法马拉松 23】谷歌的恐龙
http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1765 设答案为\(X\). 则\[X=\frac{m}{n}\times ...
- 随便玩玩系列之一:SPOJ-RNG+51nod 算法马拉松17F+51nod 1034 骨牌覆盖v3
先说说前面的SPOJ-RNG吧,题意就是给n个数,x1,x2,...,xn 每次可以生成[-x1,x1]范围的浮点数,把n次这种操作生成的数之和加起来,为s,求s在[A,B]内的概率 连续形的概率 假 ...
- 51NOD 算法马拉松8
题目戳这里:51NOD算法马拉松8 某天晚上kpm在玩OSU!之余让我看一下B题...然后我就被坑进了51Nod... A.还是01串 水题..怎么乱写应该都可以.记个前缀和然后枚举就行了.时间复杂度 ...
- 51nod 算法马拉松 34 Problem D 区间求和2 (FFT加速卷积)
题目链接 51nod 算法马拉松 34 Problem D 在这个题中$2$这个质数比较特殊,所以我们先特判$2$的情况,然后仅考虑大于等于$3$的奇数即可. 首先考虑任意一个点对$(i, j)$ ...
- 51Nod 算法马拉松21(迎新年)
这次打算法马拉松是在星期五的晚上,发挥还算正常(废话,剩下的题都不会= =). 讲讲比赛经过吧. 8:00准时发题,拿到之后第一时间开始读. A配对,看上去像是二分图最大权匹配,一看范围吓傻了,先跳过 ...
- 推荐一个算法编程学习中文社区-51NOD【算法分级,支持多语言,可在线编译】
最近偶尔发现一个算法编程学习的论坛,刚开始有点好奇,也只是注册了一下.最近有时间好好研究了一下,的确非常赞,所以推荐给大家.功能和介绍看下面介绍吧.首页的标题很给劲,很纯粹的Coding社区....虽 ...
- 51Nod 算法马拉松15 记一次悲壮而又开心的骗分比赛
OwO 故事的起源大概是zcg前天发现51Nod晚上有场马拉松,然后他就很开心的过去打了 神奇的故事就开始了: 晚上的时候我当时貌似正在写线段树?然后看见zcg一脸激动告诉我第一题有九个点直接输出B就 ...
- 51Nod 算法马拉松22 开黑记
这是一场惨烈的开黑大战,始于全机房开黑指望刷进rank前十拿钱的壮志,终于被各路神犇怒踩成rank20,差点200点头盾不保的落魄,想起将近一年前ad和zcg等学长挤进rank10的壮举,不由得唏嘘, ...
随机推荐
- 【BZOJ】2331: [SCOI2011]地板 插头DP
[题意]给定n*m的地板,有一些障碍格,要求用L型的方块不重不漏填满的方案数.L型方块是从一个方格向任意两个相邻方向延伸的方块,不能不延伸.n*m<=100. [算法]插头DP [题解]状态0表 ...
- js刷题:leecode 25
原题:https://leetcode.com/problems/reverse-nodes-in-k-group/ 题意就是给你一个有序链表.如1->2->3->4->5,还 ...
- 从ZoomEye API 到 Weblogic 弱口令扫描
参考资料: ZoomEye API: https://www.zoomeye.org/api/doc Weblogic-Weakpassword-Scnner: https://github.com/ ...
- 1、编写第一个java程序--Hello—World
1.下载JDK8.0文件 下载网址:https://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.ht ...
- 蓝色的企业后台cms管理系统——后台
链接:http://pan.baidu.com/s/1kViBtTt 密码:7hbk
- php pdo封装类
class MYPDO { protected static $_instance = null; protected $dbname = ''; protected $dsn; protected ...
- RSA加密登录
1.首先下载前端JS加密框架:jsencrypt 2.后台添加解密帮助类:RSACrypto(参考文章最后) 3.在登录页面先引入jquery.min.js,在引入jsencrypt.min.js 4 ...
- Photon3Unity3D.dll 解析二——EventData
EventData 包含Photon事件的所有内容 Code 用于表示事件,相当于主键ID,LiteEventCode定义了一部分服务端普遍事件事件: Parameters 事 ...
- 从输入URL到显示页面的过程分析
作为一个软件开发者,你一定会对网络应用如何工作有一个完整的层次化的认知,同样这里也包括这些应用所用到的技术:像浏览器,HTTP,HTML,网络服务器,需求处理等等. 本文将更深入的研究当你输入一个网址 ...
- Redis学习-redis概述
最近刚刚接触了redis技术,对此有一些了解,这是简单做一点总结. Redis简介 首先,简单了解一下NoSQL(Not only sql),不要错误的理解为:没有SQL,而是不仅仅是SQL.NoSQ ...