【做题】agc003E - Sequential operations on Sequence——经典结论
题意:有一个序列,初始是从\(1\)到\(n\)的\(n\)个数。有\(q\)次操作,每次操作给出\(q_i\),把当前的序列重复无数遍,然后截取最前面的\(q_i\)个元素作为新序列。要求输出完成所有\(q\)次操作后,每个\(1\)到\(n\)的数各出现了多少次。
\(n,q \leq 10^5, \, q_i \leq 10^{18}\)
首先一个要点在于操作后序列的长度可以达到\(10^{18}\)。因此,要从每次操作间寻找思路。
首先,我们发现如果一次操作之后又\(q_i\)比它小的操作,那么它就是没有意义的。所以,我们可以让\(q_i\)单调递增。然后,考虑每一次操作后的结果,它就是前一次操作后的结果重复若干遍,再加上前一次操作的一个前缀。前面一部分可以通过计算前一次操作后的结果时乘上一个系数来处理。那么,问题就变成如何后面的不完整部分了。
更确切地说,我们的答案还要加上一次操作后结果的一个前缀,其长度是\(q_i \bmod p_{i-1}\)。这里敏感的人或许已经发现了,我们又一个经典的结论,就是一个数对一个不大于它的数取模后,至少减小一倍。证明显然。那么,我们每次二分小一个不大于它的\(q_i\),最多做\(O(\log n)\)次后就能把这个长度缩小到\(n\)以内。到最后,就是答案序列的一个前缀加了。这里的复杂度是\(O(\log^2 n)\)。
因此,我们就能在\(O(n \log^2 n)\)时间内解决本题。
#include <bits/stdc++.h>
#define int long long
using namespace std;
const int N = 100010;
int q,n,len[N],val[N],ans[N],m;
void doit(int x,int v) {
if (!x) return;
int t = upper_bound(len+1,len+m+1,x) - len - 1;
if (t == 0) ans[1] += v, ans[x+1] -= v;
else {
val[t] += (x / len[t]) * v;
doit(x % len[t],v);
}
}
signed main() {
int x;
scanf("%lld%lld",&n,&q);
len[++m] = n;
for (int i = 1 ; i <= q ; ++ i) {
scanf("%lld",&x);
while (m && len[m] >= x) -- m;
len[++m] = x;
}
val[m] = 1;
for (int i = m ; i >= 2 ; -- i) {
val[i-1] += val[i] * (len[i] / len[i-1]);
doit(len[i] % len[i-1],val[i]);
}
ans[1] += val[1];
ans[len[1]+1] -= val[1];
for (int i = 2 ; i <= n ; ++ i)
ans[i] += ans[i-1];
for (int i = 1 ; i <= n ; ++ i)
printf("%lld\n",ans[i]);
return 0;
}
小结:还是不够熟练。以及,想题时用笔整理思路,或许是个好习惯。
【做题】agc003E - Sequential operations on Sequence——经典结论的更多相关文章
- agc003E Sequential operations on Sequence
题意: 有一个数字串S,初始长度为n,是1 2 3 4 …… n. 有m次操作,每次操作给你一个正整数a[i],你先把S无穷重复,然后把前a[i]截取出来成为新的S. 求m次操作后,每个数字在S中出现 ...
- 【agc003E】Sequential operations on Sequence
Portal -->agc003E Description 给你一个数串\(S\),一开始的时候\(S=\{1,2,3,...,n\}\),现在要对其进行\(m\)次操作,每次操作给定一个\(a ...
- Agc003_E Sequential operations on Sequence
传送门 题目大意 $1,2...n,n$个数从小到大排列,有$m$此操作,每次操作给定一个参数$x$,将当且数列作为循环节无限地展开下去,再取前$x$个作为新的数列,求最终的数列每个数出现的次数. $ ...
- 【AGC003 E】Sequential operations on Sequence
Description 你有一个长度为 \(n\) 的序列,第 \(i\) 项为 \(i\). 有 \(m\) 次操作,每次操作给定一个 \(x\),你需要将序列无限循环后截取前 \(x\) 项,作为 ...
- AtCoder Grand Contest 003 E - Sequential operations on Sequence
题目传送门:https://agc003.contest.atcoder.jp/tasks/agc003_e 题目大意 一串数,初始为\(1\sim N\),现有\(Q\)个操作,每次操作会把数组长度 ...
- (Step1-500题)UVaOJ+算法竞赛入门经典+挑战编程+USACO
http://www.cnblogs.com/sxiszero/p/3618737.html 下面给出的题目共计560道,去掉重复的也有近500题,作为ACMer Training Step1,用1年 ...
- (转)poj算法做题顺序
初期: 一.基本算法: (1)枚举. (poj1753,poj2965) (2)贪心(poj1328,poj2109,poj2586) (3)递归和分治法. (4)递推. (5)构造法.(poj329 ...
- [日记&做题记录]-Noip2016提高组复赛 倒数十天
写这篇博客的时候有点激动 为了让自己不颓 还是写写日记 存存模板 Nov.8 2016 今天早上买了两个蛋挞 吃了一个 然后就做数论(前天晚上还是想放弃数论 但是昨天被数论虐了 woc noip模拟赛 ...
- AtCoder Grand Contest 1~10 做题小记
原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-1-to-10.html 考虑到博客内容较多,编辑不方便的情 ...
随机推荐
- Rpgmakermv(24 )yep_coreengine
==左部为原文,右边我做了简要翻译=================================== Introduction and Instructions ================= ...
- 31网络通信之Select模型
多路复用并发模型 -- select #include<sys/select.h> #include<sys/time.h> int select(int maxfd, f ...
- JavaScript循环和数组常用操作
while循环 语法: do while循环 语法:do{循环体}while(条件表达式); 特点:do while循环不管条件是否成立,无论如何循环体都会执行一次. 使用场合:用户输入密码,如果密码 ...
- Robot Framework 遇到过的错误 1. Chrome打开无法数据网址,地址栏只显示data:,
问题描述:用RF打开网页时未跳转到指定网址,而是显示data:, *** Settings ***Library SeleniumLibrary *** Test Cases ***Login_Tes ...
- LinkedList 底层实现原理
LinkedList的底层实现原理 LinkedList 底层数据结构为双向链表,链表结构,基于一个个链表节点Node 1,Inner Class 内部类 private static class N ...
- 20165305 实验四:Android程序设计
实验内容 基于Android Studio开发简单的Android应用并部署测试; 了解Android.组件.布局管理器的使用: 掌握Android中事件处理机制. Android Studio安装 ...
- 设计模式之Observer(观察者)(转)
Java深入到一定程度,就不可避免的碰到设计模式(design pattern)这一概念,了解设计模式,将使自己对java中的接口或抽象类应用有更深的理解.设计模式在java的中型系统中应用广泛,遵循 ...
- 大数据和hadoop有什么关系?
本文资料来自百度文库相关文档 Hadoop,Spark和Storm是目前最重要的三大分布式计算系统,Hadoop常用于离线的复杂的大数据处理,Spark常用于离线的快速的大数据处理,而Storm常用于 ...
- 转:C#判断ContextMenuStrip右键菜单的来源(从哪个控件弹出来的)
转载自:http://hi.baidu.com/cookiemulan/item/82df8ff867dd53cc531c26c7 有时候,为了提高性能和节约资源,我们会为多个控件,指定同一个右键弹出 ...
- 常见的原生javascript DOM操作
1.创建元素 创建元素:document.createElement() 使用document.createElement()可以创建新元素.这个方法只接受一个参数,即要创建元素的标签名.这个标签名在 ...