CF837F - Prefix Sums
首先,我们发现这道题目“序列会增长”的情况完全就是唬人的,因为我们把 \(x_i\) 输入之后,\(y_i\) 永远是 \(0\),而前导 \(0\) 在计算的过程中没有任何的作用。所以可以直接原地做前缀和。
我们还发现,除了序列 \(A^0\),其他的序列都是递增的(\(A^i_j-A^i_{j-1}=A^{i-1}_{j}\ge 0\)),所以 \(s\gt 0\) 时,序列 \(A^s\) 存在 \(A^s_i\ge k\) 等价于 \(A^s_n\ge k\)。
然后,我们考虑二分答案,每次检测进行 \(a\) 次运算之后的序列是否满足条件。
首先考虑的是生成函数和卷积,因为如果我们构造生成函数 \(f(x)=A^0_1x+A^0_2x^2+\cdots\) 以及 \(\mathrm{I}(x)=1+x+x^2+x^3+\cdots\),那么求 \(a\) 次前缀和的结果就是 \(f(x)\times(\mathrm{I}(x))^a\)。直接看第 \(n\) 位即可。求 \((\mathrm{I}(x))^a\) 可以使用卷积意义下的快速幂进行,复杂度 \(O(n\log n\log a)\),看上去可以过,但别忘了,我们这是在二分答案的 check
,所以总复杂度还要多一个 \(\log k\),铁定过不了。(而且系数和 \(k\) 取 \(\min\) 也可能达到 \(10^{18}\) 级别,点值表示后其大小和 complex
误差不是 FFT 配 __int128
所能承受的)
我们先把序列所有的前导 \(0\) 去掉,找到序列第一个 \(1\) 的位置,考虑它对答案的贡献。我们发现,只要序列的长度达到一定长度,只有开头一个 \(1\) 的序列的第 \(n\) 项就会以指数级增长(即 \(a\) 次运算后,第 \(n\) 项的级别是 \(O(n^a)\)。虽然我们知道这可能有一个很小的常数,但是不要紧,当 \(n\ge 10\) 的时候,达到 \(10^{18}\) 的时间也不会超过 \(500\)。更何况 \(a\) 会随着 \(n\) 的增大而减小,所以我们可以直接暴力做。
对于 \(n\lt 10\),我们考虑别的方法。
常见的思路是 \(O(n^3\log^2 k)\) 的二分答案和矩阵快速幂。但是我们前面已经思考了生成函数做法,不如将其推到底。
首先,直接暴力做卷积是可以做的,但是这就显得很怨种。我们都已经推到 \(n\lt 10\) 了,再用多项式总是有点杀鸡用牛刀的感觉(而且点值表示后虽然不会爆 __int128
了,误差还是有点可怕)。
我们考虑看看 \(f(x)\times(\mathrm{I}(x))^a\) 这个式子,发现 \(A^0_i\) 对答案 \((f\times\mathrm{I}^a)(x)\) 的第 \(n\) 项的贡献是 \(\mathrm{I}^{a-1}(x)\) 的第 \(n-i\) 项系数。
考虑求出这个东西,\(G(x)=\mathrm{I}^{a-1}(x)=\dfrac{1}{(1-x)^{a-1}}\),求第 \(y\) 项就是求 \(\dfrac{G^{(y)}(0)}{y!}\)。而 \(G^{(y)}(x)=\dfrac{(a-1)\cdots(a+y-2)}{(1-x)^{a-1+y}}\),则 \(\dfrac{G^{(y)}(0)}{y!}=\dfrac{(a+y-2)!}{(1-0)^{a-1+y}(a-2)!y!}={a+y-2 \choose y}\)
然后我们的 \(y=n-i\),所以需要的 \((f\times\mathrm{I}^a)(x)\) 的第 \(n\) 项其实就是 \(\sum_{i\le n}{{a+n-i-2\choose n-i+1}A^0_i}\)。
接下来,我们发现 \(n-i+1\) 很小,我们如果记 \(p=n-i+1,q=a+n-i-2\),这个组合数就是 \(\dfrac{p(p-1)(p-2)\cdots (p-q+1)}{q!}\),而上下都不超过 \(n\) 项,下面的 \(q!\) 不超过 \(3628800\),所以我们可以先暴力做出 \(q!\),然后用一个 __int128
暴力维护上面的乘积。一旦这个积的大小超过了 \(q!\times k\),就直接返回 \(k\) 退出。然后扫 \(1\) 到 \(n\),计算出第 \(n\) 项的答案,中途大于 \(k\) 就直接返回 \(1\)。当然我们记住我们是在二分答案的,组合数的计算也是 \(O(n)\) 的,\(n\lt 10\) 的部分复杂度是 \(O(n^2\log k)\)。
一些闲话:不会生成函数怎么推最后的组合数式子?
我们可以只看第 \(i\) 项,考虑它对答案的贡献。实际上就是抹掉前 \(i-1\) 项,贡献次数就是对一个 \(1\) 和 \(n-i\) 个 \(0\) 做 \(a\) 次前缀和之后的第 \(n-i+1\) 项。
我们发现,如果我们把 1 1 1 1 1 1 1...
这样的无穷序列排下来,求 \(x\) 次前缀和之后的第 \(y\) 项的话,其实就是在表 \(S_{i,j}=S_{i-1,j}+S_{j-1,i}\) 中求第 \(x\) 行的第 \(j\) 列。
如果我们把这个表列出来:
1 | 1 1 1 1 1
2 | 1 2 3 4 5
3 | 1 3 6 10 15
4 | 1 4 10 20 35
如此列下来,你会发现,它就是旋转 \(45\) 度之后的杨辉三角!如果我们把左下到右上对角线看作一行的话,\(S_{i,j}=S_{i-1,j}+S_{j-1,i}\) 正好满足组合数的递推形式。
那么,我们进行坐标变换,容易发现第 \(x\) 行的第 \(y\) 列(从 \(1\) 开始)对应到杨辉三角坐标之后,处在第 \(x+y-2\) 行的第 \(y-1\) 列(从 \(0\) 开始)。而此处的系数就是 \({x+y-2\choose y-1}\),代入得到 \({a+n-i-2\choose n-i+1}\)。也就使用非生成函数的办法推出了这个式子。
#define rp(i,n) for(ll i=1;i<=n;i++)
#define rep(i,a,b) for(ll i=a;i<=b;i++)
ll n,k,a[200005];
__int128 A[200005];
inline __int128 C(ll n,ll m){
__int128 res=1,f=1;
rp(i,m)f*=i;
rep(i,n-m+1,n){
res*=i;
if(res>=f*k)return k;
}
return res/f;
}
inline void solve1(){
ll mx=0;
rp(i,n)A[i]=a[i];
int cur=0;
while(A[n]<k){
rp(i,n)A[i]=A[i-1]+A[i];
cur++;
}
cout<<cur<<endl;
}
inline bool check(ll f){
__int128 ans=0;
if(f>=k)return 1;
rp(i,n){
ll x=f,y=n-i+1;
__int128 dd=C(x+y-2,y-1);
if(a[i]&&dd>=k)return 1;
ans+=a[i]*dd;
if(ans>=k)return 1;
}
return ans>=k;
}
inline void solve2(){
ll l=1,r=1e18,mid,ans=0;
while(l<=r){
mid=l+r>>1;
if(check(mid))ans=mid,r=mid-1;
else l=mid+1;
}cout<<ans<<endl;
}
signed main(){
ios::sync_with_stdio(false);
cin.tie(0);
cin>>n>>k;
rp(i,n)cin>>a[i];
rp(i,n)if(a[i]>=k){
cout<<0<<endl;
return 0;
}
int be=1;
while(!a[be])be++;be--;
rp(i,n-be)a[i]=a[i+be];n-=be;
if(n>10)solve1();
else solve2();
return 0;
}
//Crayan_r
CF837F - Prefix Sums的更多相关文章
- 【题解】【数组】【Prefix Sums】【Codility】Genomic Range Query
A non-empty zero-indexed string S is given. String S consists of N characters from the set of upper- ...
- 【题解】【数组】【Prefix Sums】【Codility】Passing Cars
A non-empty zero-indexed array A consisting of N integers is given. The consecutive elements of arra ...
- Codeforces 837F Prefix Sums
Prefix Sums 在 n >= 4时候直接暴力. n <= 4的时候二分加矩阵快速幂去check #include<bits/stdc++.h> #define LL l ...
- CodeForces 837F - Prefix Sums | Educational Codeforces Round 26
按tutorial打的我血崩,死活挂第四组- - 思路来自FXXL /* CodeForces 837F - Prefix Sums [ 二分,组合数 ] | Educational Codeforc ...
- Educational Codeforces Round 26 [ D. Round Subset ] [ E. Vasya's Function ] [ F. Prefix Sums ]
PROBLEM D - Round Subset 题 OvO http://codeforces.com/contest/837/problem/D 837D 解 DP, dp[i][j]代表已经选择 ...
- CodeForces 1204E"Natasha, Sasha and the Prefix Sums"(动态规划 or 组合数学--卡特兰数的应用)
传送门 •参考资料 [1]:CF1204E Natasha, Sasha and the Prefix Sums(动态规划+组合数) •题意 由 n 个 1 和 m 个 -1 组成的 $C_{n+m} ...
- CF1303G Sum of Prefix Sums
点分治+李超树 因为题目要求的是树上所有路径,所以用点分治维护 因为在点分治的过程中相当于将树上经过当前$root$的一条路径分成了两段 那么先考虑如何计算两个数组合并后的答案 记数组$a$,$b$, ...
- GenomicRangeQuery /codility/ preFix sums
首先上题目: A DNA sequence can be represented as a string consisting of the letters A, C, G and T, which ...
- codeforces:Prefix Sums分析和实现
题目大意: 给出一个函数P,P接受一个数组A作为参数,并返回一个新的数组B,且B.length = A.length + 1,B[i] = SUM(A[0], ..., A[i]).有一个无穷数组序列 ...
- [CF1204E]Natasha,Sasha and the Prefix Sums 题解
前言 本文中的排列指由n个1, m个-1构成的序列中的一种. 题目这么长不吐槽了,但是这确实是一道好题. 题解 DP题话不多说,直接状态/变量/转移. 状态 我们定义f表示"最大prefix ...
随机推荐
- 【大数据面试】sqoop:空值、数据一致性、列式存储导出、数据量、数据倾斜
一.有没有遇到过问题,怎么进行解决的 1.空值问题 本质:hive底层存储空数据使用\n<==>MySQL存储空数据使用null 解决:双向导入均分别使用两个参数☆,之前讲过 2.数据一致 ...
- 真实世界的人工智能应用落地——OpenAI篇 ⛵
作者:韩信子@ShowMeAI 深度学习实战系列:https://www.showmeai.tech/tutorials/42 本文地址:https://www.showmeai.tech/artic ...
- [OpenCV实战]7 使用YOLOv3和OpenCV进行基于深度学习的目标检测
目录 1 YOLO介绍 1.1 YOLOv3原理 1.2 为什么要将OpenCV用于YOLO? 1.3 在Darknet和OpenCV上对YOLOv3进行速度测试 2 使用YOLOv3进行对象检测(C ...
- dfs学习笔记
题目链接 可以通过参考一道例题来加深对dfs的认知和学习 题意描述 按照字典序输出自然数 1 到 n 所有不重复的排列,即 n 的全排列,要求所产生的任一数 字序列中不允许出现重复的数字. 输出格式 ...
- vulnhub靶场之FUNBOX: UNDER CONSTRUCTION!
准备: 攻击机:虚拟机kali.本机win10. 靶机:Funbox: Under Construction!,下载地址:https://download.vulnhub.com/funbox/Fun ...
- Linux C 用GPS时间更新系统时间的方法。
思路: 1.GPS模块会自动收到带时间信息的消息. GPS模块会收到很多的协议消息带时间信息的.我们选择"$GPRMC"这条协议.其中的时间格式有的是hhmmss(时分秒) 有的是 ...
- 洛谷P1862输油管道问题
题目传送门 二话不说先上代码: #include<bits/stdc++.h> using namespace std; int n; int x,y[10001]; int main() ...
- appium如何连接多台设备
我们在做app自动化的时候,若要考虑兼容性问题,需要跑几台设备,要是一台一台的跑比较耗 时,因此需要考虑使用多线程来同时操作多台设备. 1.我们拿两台设备来模拟操作下,使用:adb devices查看 ...
- CDH-hive内进行删除操作
hive安装后需要修改已建的表及查询操作,在执行修改操作时遇到了如下问题. hive> update dp set name='beijing' where id=1159; FAILED: ...
- sstream中的stringstream怎么用
sstream中的stringstream怎么用 1.cin cin是从缓冲区读入,会把空格.回车等不可见的字符当做是分割,跳过.并且最后读入之后,后面会有剩余的部分,比如空格.回车等. 2.getl ...