\(\mathcal{Description}\)

  \(n\) 个点,第 \(i\) 个点能走向第 \(d_i\) 个点,但从一个点出发至多走 \(k\) 步。对于每个点,求有多少点能够走到它。

  \(n\le5\times10^5\)。

\(\mathcal{Solution}\)

  显然这些点构成一片内向基环树森林。考虑每个点的贡献,若其向上走 \(k\) 步仍不能到环上,那一定只在树内对一条链贡献,树上差分一下。否则,该点会向到根的每个点贡献,并向环上一段连续的点贡献,后者维护一个数列差分亦可统计,就切掉啦。

  复杂度 \(\mathcal O(n)\)。

\(\mathcal{Code}\)

#include <cstdio>
#include <vector>
#include <algorithm> inline int rint () {
int x = 0; char s = getchar ();
for ( ; s < '0' || '9' < s; s = getchar () );
for ( ; '0' <= s && s <= '9'; s = getchar () ) x = x * 10 + ( s ^ '0' );
return x;
} inline void wint ( const int x ) {
if ( 9 < x ) wint ( x / 10 );
putchar ( x % 10 ^ '0' );
} const int MAXN = 5e5;
int n, K, d[MAXN + 5], ans[MAXN + 5], stk[MAXN + 5], top, ccnt;
bool vis[MAXN + 5], oncir[MAXN + 5], inq[MAXN + 5];
int dep[MAXN + 5], fa[MAXN + 5], dir[MAXN + 5], tag[MAXN + 5], add[MAXN * 2 + 5];
std::vector<int> cir[MAXN + 5], adj[MAXN + 5]; inline void findCir ( const int u ) {
if ( vis[u] ) return ;
vis[u] = true, stk[++ top] = u, inq[u] = true;
if ( inq[d[u]] ) {
int v; ++ ccnt;
do {
oncir[v = stk[top --]] = true;
cir[ccnt].push_back ( v );
} while ( v ^ d[u] );
std::reverse ( cir[ccnt].begin (), cir[ccnt].end () );
}
findCir ( d[u] ), inq[u] = false;
} inline void solveTree ( const int u, int p, const int rtid, const int L ) {
++ tag[u];
if ( dep[u] <= K ) {
int tar = K - dep[u] + 1;
if ( tar >= L ) tar = L - 1;
++ add[rtid + 1], -- add[rtid + 1 + tar];
} else {
-- tag[fa[p]], p = dir[p];
}
for ( int i = 0, v; i ^ adj[u].size (); ++ i ) {
if ( !oncir[v = adj[u][i]] ) {
dep[dir[u] = v] = dep[u] + 1, fa[v] = u;
solveTree ( v, p, rtid, L );
}
}
} inline int calc ( const int u ) {
int ret = tag[u];
for ( int i = 0, v; i ^ adj[u].size (); ++ i ) {
if ( !oncir[v = adj[u][i]] ) {
ret += calc ( v );
}
}
return ans[u] += ret, ret;
} inline void solveCir ( std::vector<int>& cir ) {
int L = cir.size ();
for ( int i = 0; i <= L << 1; ++ i ) add[i] = 0;
for ( int i = 0, u; i < L; ++ i ) {
dep[u = cir[i]] = 1;
solveTree ( u, u, i, L ), calc ( u );
}
for ( int i = 1; i < L * 2; ++ i ) add[i] += add[i - 1];
for ( int i = 0; i < L; ++ i ) ans[cir[i]] += add[i] + add[i + L];
} int main () {
freopen ( "travel.in", "r", stdin );
freopen ( "travel.out", "w", stdout );
n = rint (), K = rint ();
for ( int i = 1; i <= n; ++ i ) adj[d[i] = rint ()].push_back ( i );
for ( int i = 1; i <= n; ++ i ) top = 0, findCir ( i );
for ( int i = 1; i <= ccnt; ++ i ) solveCir ( cir[i] );
for ( int i = 1; i <= n; ++ i ) wint ( ans[i] ), putchar ( '\n' );
return 0;
}

\(\mathcal{Details}\)

  考场上想得久了点 qwq……最神奇的是兔子爆排一般是因为标算写假而测试代码是对的。(

Solution -「LOCAL」人口迁徙的更多相关文章

  1. Solution -「LOCAL」二进制的世界

    \(\mathcal{Description}\)   OurOJ.   给定序列 \(\{a_n\}\) 和一个二元运算 \(\operatorname{op}\in\{\operatorname{ ...

  2. Solution -「LOCAL」大括号树

    \(\mathcal{Description}\)   OurTeam & OurOJ.   给定一棵 \(n\) 个顶点的树,每个顶点标有字符 ( 或 ).将从 \(u\) 到 \(v\) ...

  3. Solution -「LOCAL」过河

    \(\mathcal{Description}\)   一段坐标轴 \([0,L]\),从 \(0\) 出发,每次可以 \(+a\) 或 \(-b\),但不能越出 \([0,L]\).求可达的整点数. ...

  4. Solution -「LOCAL」Drainage System

    \(\mathcal{Description}\)   合并果子,初始果子的权值在 \(1\sim n\) 之间,权值为 \(i\) 的有 \(a_i\) 个.每次可以挑 \(x\in[L,R]\) ...

  5. Solution -「LOCAL」Burning Flowers

      灼之花好评,条条生日快乐(假装现在 8.15)! \(\mathcal{Description}\)   给定一棵以 \(1\) 为根的树,第 \(i\) 个结点有颜色 \(c_i\) 和光亮值 ...

  6. Solution -「LOCAL」画画图

    \(\mathcal{Description}\)   OurTeam.   给定一棵 \(n\) 个点的树形随机的带边权树,求所有含奇数条边的路径中位数之和.树形生成方式为随机取不连通两点连边直到全 ...

  7. Solution -「LOCAL」ZB 平衡树

    \(\mathcal{Description}\)   OurOJ.   维护一列二元组 \((a,b)\),给定初始 \(n\) 个元素,接下来 \(m\) 次操作: 在某个位置插入一个二元组: 翻 ...

  8. Solution -「LOCAL」舟游

    \(\mathcal{Description}\)   \(n\) 中卡牌,每种三张.对于一次 \(m\) 连抽,前 \(m-1\) 次抽到第 \(i\) 种的概率是 \(p_i\),第 \(m\) ...

  9. Solution -「LOCAL」充电

    \(\mathcal{Description}\)   给定 \(n,m,p\),求序列 \(\{a_n\}\) 的数量,满足 \((\forall i\in[1,n])(a_i\in[1,m])\l ...

随机推荐

  1. BIO、NIO、AIO --- 个人理解

    1.前言 什么是 BIO.NIO.AIO  ,不难看出,都是共同的字符IO , IO的意思是input output  ,即输入输出 , 那么 B . N .A 分别指不同的io模型 ,而io又分为 ...

  2. lua中的三目运算符

    开头先说结论 1.简单版三目运算符(需要自我保证"b"不为"false") a and b or c 2.通用版三目运算符 (a and {b} or {c}) ...

  3. Keil MDK STM32系列(三) 基于标准外设库SPL的STM32F407开发

    Keil MDK STM32系列 Keil MDK STM32系列(一) 基于标准外设库SPL的STM32F103开发 Keil MDK STM32系列(二) 基于标准外设库SPL的STM32F401 ...

  4. rocketmq实现延迟队列精确到秒级实现(总结编)

    前言篇: 为了节约成本,决定通过自研来改造rocketmq,添加任意时间延迟的延时队列,开源版本的rocketmq只有支持18个等级的延迟时间, 其实对于大部分的功能是够用了的,但是以前的项目,全部都 ...

  5. [USB波形分析] 全速USB波形数据分析(三)

    前面的两篇文章介绍和分析了USB的一些基本知识,结合前面的介绍,今天用实例介绍USB的枚举过程. 1 | 概况 硬件基于EK-TMC123GXL开发板,软件是TI提供的USB批量传输的简单例子,在PC ...

  6. [µC/GUI 学习]µC/GUI移植

    一.什么是µC/GUI µC/GUI为任何需要图形显示器的嵌入式应用提供了一种灵活的图形用户界面(GUI).µC/GUI允许软件工程师在使用了LCD显示器的产品上增加美轮美奂的用户界面,从简单的2D黑 ...

  7. Cesium中级教程8 - Introduction to Particle Systems 粒子系统入门

    Cesium中文网:http://cesiumcn.org/ | 国内快速访问:http://cesium.coinidea.com/ What is a particle system? 什么是粒子 ...

  8. vue3源码node的问题

    下载vue3源码后,下载依赖时,node的版本需要在10.0.0以上,并且不同的vue3里面的插件的配置对版本依赖还不同,14.0.0以上的版本基本都不支持win7了, win7系统可以安装12.0. ...

  9. Natasha 4.0 探索之路系列(三) 基本的动态编译

    Natasha 的设计 动态编译 Roslyn 为开发者提供了动态编译的接口, 允许我们以 C# 代码来编写 Emit 或 表达式树生成的程序集, 但是完成一个编译需要诸多步骤, 用户参与的操作也很多 ...

  10. 虚拟化技术kvm,xen,vmware比较

    目前市面上常用的虚拟机技术主要有KVM.xen.vmware. KVM是指基于Linux内核(Kernel-based)的虚拟机(Virtual Machine).KVM最大的好处就在于它是与Linu ...