首先我们可以考虑一下 \(x\) 关于 \(y\) 的对称点的坐标,不难发现就是 \(x + 2 \times (y - x)\),那么期望的增量就会增加 \(2 \times (y - x)\)。不难发现我们可以令 \(dp_{i, j}\) 表示 \(j\) 第 \(i\) 次操作(即假设进行了 \(p\) 轮操作,当前是第 \(q\) 次操作,则 \(i = (p - 1) \times m + q\)),则会有转移:

\[dp_{i, a_q} += \frac{1}{2} \times 2 \times (dp_{i - 1, a_q - 1} - dp_{i - 1, a_q}) + \frac{1}{2} \times 2 \times (dp_{i - 1, a_q + 1} - dp_{i - 1, a_q})
\]

即:

\[dp_{i, a_q} += dp_{i - 1, a_q - 1} + dp_{i - 1, a_q + 1} - 2 \times dp_{a_q}
\]

可以发现第一维可以直接压去,则有转移 \(i = a_q\):

\[dp_i += dp_{i - 1} + dp_{i + 1} - 2 \times dp_i
\]

你会发现这个转移非常有特色,实际上可以将上式划成两个部分:

\[dp_i += (dp_{i + 1} - dp_i) - (dp_i - dp_{i - 1})
\]

你会发现这实际上是两个差分数在相加,令 \(f_i = dp_i - dp_{i - 1}\),即:

\[dp_i = f_{i + 1} - f_i
\]

显然这样算是非常不方便的,我们干脆考虑差分数组 \(f\) 的变化,则有:

\[f_i += f_{i + 1} - f_i \Leftrightarrow f_i = f_{i + 1}
\]
\[f_{i + 1} -= f_{i + 1} - f_i \Leftrightarrow f_{i + 1} = f_i
\]

相当于将差分数组 \(f_i, f_{i + 1}\) 交换!那么问题转化为,给你一个序列,有 \(K\) 轮变换,每次变换有 \(m\) 次操作,按照给定的操作序列 \(a_i\) 交换 \(f_{a_i}, f_{a_i + 1}\)。因为每轮变换的操作序列都是一样的,这意味着每个位置在一轮之后变到的位置是一样的。于是我们先暴力将经过一轮操作后位置排列 \(P\) 变换到的位置序列 \(P'\)(事实上我们成排列 \(P \rightarrow P'\) 的这样一个过程叫做置换),对于置换而言一个常见的想法就是对 \(P_i \rightarrow P'_i\) 连一条边,这样将形成一张若干个简单环组成的有向图。那么每个位置经过 \(K\) 轮置换后的位置就相当于它在这个简单环上走 \(K\) 步后的位置,这个就很容易可以算出来了。

可以发现,上面这个做法的复杂度是 \(O(n + m)\) 的,非常地优秀!

#include<bits/stdc++.h>
using namespace std;
#define int long long
#define rep(i, l, r) for(int i = l; i <= r; ++i)
const int N = 100000 + 5;
bool book[N];
int n, m, k, tmp, cnt, a[N], d[N], p[N], dp[N], nxt[N], ans[N];
vector <int> G[N];
int read(){
char c; int x = 0, f = 1;
c = getchar();
while(c > '9' || c < '0'){ if(c == '-') f = -1; c = getchar();}
while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return x * f;
}
signed main(){
n = read();
rep(i, 1, n) d[i] = read(), p[i] = i, dp[i] = d[i] - d[i - 1];
m = read(), k = read();
rep(i, 1, m) a[i] = read(), swap(p[a[i]], p[a[i] + 1]);
rep(i, 1, n) nxt[i] = p[i];
rep(i, 1, n) if(!book[i]){
int x = i; G[++cnt].push_back(i), book[i] = true;
while(nxt[x] != i) x = nxt[x], book[x] = true, G[cnt].push_back(x);
}
rep(i, 1, cnt){
int S = G[i].size(), tr = k % S;
rep(j, 0, S - 1) ans[G[i][j]] = dp[G[i][(j + tr) % S]];
}
rep(i, 1, n) tmp += ans[i], printf("%lld.0\n", tmp);
return 0;
}

AT2164 [AGC006C] Rabbit Exercise的更多相关文章

  1. [AT2164] [agc006_c] Rabbit Exercise

    题目链接 AtCoder:https://agc006.contest.atcoder.jp/tasks/agc006_c 洛谷:https://www.luogu.org/problemnew/sh ...

  2. 【做题】agc006C - Rabbit Exercise——模型转换

    原文链接https://www.cnblogs.com/cly-none/p/9745177.html 题意:数轴上有\(n\)个点,从\(1\)到\(n\)编号.有\(m\)个操作,每次操作给出一个 ...

  3. AGC006C Rabbit Exercise

    传送门 设 \(f_{i,j}\) 表示兔子 \(i\) 在当前 \(j\) 轮的期望位置 对于一次操作 \(f_{i,j+1}=\frac{1}{2}(2f_{i-1,j}-f_{i,j})+\fr ...

  4. 【agc006C】Rabbit Exercise

    Portal --> agc006C Solution 啊感觉是好有意思的一道题qwq官方题解里面的说辞也是够皮的哈哈哈..(大概就是说如果你没有意识到那个trick的话这题这辈子都做不出来qw ...

  5. 【AGC006C】Rabbit Exercise 置换

    题目描述 有\(n\)只兔子站在数轴上.为了方便,将这些兔子标号为\(1\ldots n\).第\(i\)只兔子的初始位置为\(a_i\). 现在这些兔子会按照下面的规则做若干套体操.每一套体操由\( ...

  6. AT2164 Rabbit Exercise

    传送门 解题思路 首先考虑k=1的情况,对于每一个a[i],它可能会到a[i-1]*2-a[i] 与 a[i+1]*2-a[i]两个位置,概率都为%50,那么它的期望位置为 (a[i-1]*2-a[i ...

  7. 【AtCoder】【思维】【置换】Rabbit Exercise

    题意: 有n只兔子,i号兔子开始的时候在a[i]号位置.每一轮操作都将若干只兔子依次进行操作: 加入操作的是b[i]号兔子,就将b[i]号兔子移动到关于b[i]-1号兔子现在所在的位置对称的地方,或者 ...

  8. AtCoder Grand Contest 006 (AGC006) C - Rabbit Exercise 概率期望

    原文链接https://www.cnblogs.com/zhouzhendong/p/AGC006C.html 题目传送门 - AGC006C 题意 有 $n$ 个兔子,从 $1$ 到 $n$ 编号, ...

  9. [Atcoder Grand 006 C] Rabbit Exercise 解题报告 (期望)

    题目链接:https://www.luogu.org/problemnew/show/AT2164 https://agc006.contest.atcoder.jp/tasks/agc006_c 题 ...

随机推荐

  1. Reproducing Kernel Hilbert Space (RKHS)

    目录 概 主要内容 RKHS-wiki 概 这里对RKHS做一个简单的整理, 之前的理解错得有点离谱了. 主要内容 首先要说明的是, RKHS也是指一种Hilbert空间, 只是其有特殊的性质. Hi ...

  2. Capstone CS5267|CS5267参数|CS5267规格书

    CS5267 USB Type-C to HDMI2.0b 4k@60Hz Converter with PD3.0 Support 1.CS5267概述 Capstone CS5267是一款高性能T ...

  3. Windows下安装配置Maven

    1.下载Maven 官方下载地址:http://maven.apache.org/download.cgi 目前Apache Maven最小版本为3.6.3, 下载适合Windows的安装包apach ...

  4. Linux查看CPU详细信息

    1.查看CPU详细信息 在Linux服务器上查看CPU详细信息: cat /proc/cpuinfo 输出结果: processor : 0 vendor_id : GenuineIntel cpu ...

  5. C#中的隐式转换

    你是否考虑过这个问题:为什么不同类型之间的变量可以赋值,而不需要强制转换类型?如: int i = 1; long l = i; object obj = 1; Exception exception ...

  6. Linux配置yum源(离线和在线)

    配置yum源有2种方法,一种是离线yum源,另外一种是在线yum源. 一.离线yum源,基于安装光盘提供的安装仓库. 建立一个属于仓库文件夹 mkdir /media/zidong cd /media ...

  7. JMeter_请求参数

    在做接口测试时,发送请求的参数有两种格式,一种是Parameters,一种是JSON 一.Jmeter传参 JMeter 传Parameters格式的参数 JMeter 传JSON格式的参数 二.区分 ...

  8. 设置Linux的一些文本输出方式

    更新一下yum咯 yum install -y epel-release 火车 sudo yum install sl $ sl 放火 sudo yum install libaa-bin 小老鼠 s ...

  9. linux tomcat【9.0.12】 使用 ssl证书 配置 https 的具体操作 【使用 域名 】

    1.前言 根据上一个随笔,已经可以正式在 阿里云服务器发布 工程了 ,但是用的协议默认是 http ,端口80 但是 http不安全 ,容易被拦截抓包 ,于是出来了个 https tomcat发布 对 ...

  10. js箭头函数 的 (e) => { } 写法笔记

    1. (e) => {} 是ES 6 新语法,默认是Es 5.1,因此在这里设置一下就不会提示红色下划线了 2.使用: (e) => {}  , 其实就是function (e){} 的缩 ...