BZOJ4869:[SHOI2017]相逢是问候——题解
http://www.lydsy.com/JudgeOnline/problem.php?id=4869
题面复制于洛谷:https://www.luogu.org/problemnew/show/P3747#sub
参考洛谷的前两篇(也是仅有的两篇)题解。
首先我们要知道一个公式:

这又被叫做扩展欧拉定理,证明我们并不关心。
有了扩展欧拉定理,我们就能够避免高精度从而求出对于任意一个数的0操作之后变成什么数了。
(递归或者迭代选一个,递归好理解,迭代有助于理解下面的题解,而且常数小)
我们又有一个结论,对于一个p,它无限递归p=phi(p)直到p=1为止的深度为O(logp)。
这样的好处在于我们虽然修改了很多次,但是当修改次数大于logp的时候,此时你再怎么修改也没有用了因为你的指数为1相当于没有操作。
那么显然对于1我们记录该元素被操作了几次,然后暴力修改即可,可用线段树维护。复杂度O(nlognlogp)。(请注意这个复杂度是假的)
这样的复杂度我们交到bzoj上是没有问题的,但是交到洛谷上会TLE3个点。将递归改成迭代,预处理每个p的phi,各种常数优化也会TLE2个点。
emmm……why?
当然是因为我们的复杂度没算对啊。
对于单点修改,显然每次修改是O(logplogp)……等等,怎么多出来一个O(logp)。
忘了我们使用了快速幂了吗,我们多出来的O(logp)就是这么来的。
考虑除掉这个O(logp),显然预处理快速幂。
如果你写的是迭代的话,你就会发现底数永远都是c不变,变的只是指数和模数, 且指数最大是p=1e8。
我们可以先求出不同模数且指数<=1e5的c的幂,我们还可以求不同模数且指数=整1e5的c的幂。
这就很像分块了,显然当我们要求指数为k时,k=x*1e5+y(y<1e5)显然可求。
这样我们预处理出所有的数在多少次操作后的值,则我们的复杂度就是O(nlognlogp)。
吐槽:最开始学完扩欧之后觉得这题洛谷给的难度高了,怎么就NOI+了,后来在TLE之后一看woc还有这种操作……
神题神题……
(然而博主并不想写正解,放的代码只能过bzoj,正解如果有时间的话会补上的emmm)
#include<cstdio>
#include<cmath>
#include<algorithm>
#include<cstring>
#include<cctype>
#include<map>
using namespace std;
typedef long long ll;
const int N=5e4+;
const int O=1e4+;
inline int read(){
int X=,w=;char ch=;
while(!isdigit(ch)){w|=ch=='-';ch=getchar();}
while(isdigit(ch))X=(X<<)+(X<<)+(ch^),ch=getchar();
return w?-X:X;
}
struct tree{
ll v,t;
}tr[N*];
int su[O],he[O],cnt,phi[],n,m;
ll p,c,logp,b[N];
bool ok;
inline ll qpow(ll k,int p){
ll ans=,s=c;
while(k){
if(k&)ans=ans*s;
s*=s;k>>=;
if(s>=p)ok=,s%=p;
if(ans>=p)ok=,ans%=p;
}
return ans;
}
int Euler(int k){
int res=k;
for(int i=;su[i]*su[i]<=k;i++){
if(k%su[i]==){
res-=res/su[i];
while(k%su[i]==)k/=su[i];
}
}
if(k>)res-=res/k;
return res;
}
void prime(){
for(int i=;i<O;i++){
if(he[i]==){
cnt++;
su[cnt]=i;
}
for(int j=;j<=cnt&&i*su[j]<O;j++){
he[su[j]*i]=;
if(i%su[j]==)break;
}
}
phi[logp]=p;
while(phi[logp]!=)phi[++logp]=Euler(phi[logp-]);
phi[++logp]=;
}
void build(int a,int l,int r){
if(l==r){
tr[a].v=b[l]%p;
return;
}
int mid=(l+r)>>;
build(a<<,l,mid);build(a<<|,mid+,r);
tr[a].v=(tr[a<<].v+tr[a<<|].v)%p;
}
ll suan(ll v,ll k){
ll tmp=v;
if(tmp>phi[k])tmp=tmp%phi[k]+phi[k];
for(int i=k;i>;i--){
ok=;tmp=qpow(tmp,phi[i-]);
if(ok)tmp+=phi[i-];
}
return tmp;
}
void gai(int a,int l,int r,int l1,int r1){
if(tr[a].t>=logp)return;
if(r<l1||r1<l)return;
if(l==r){
tr[a].t++;
tr[a].v=suan(b[l],tr[a].t);
return;
}
int mid=(l+r)>>;
gai(a<<,l,mid,l1,r1);gai(a<<|,mid+,r,l1,r1);
tr[a].v=(tr[a<<].v+tr[a<<|].v)%p;
tr[a].t=min(tr[a<<].t,tr[a<<|].t);
}
ll wen(int a,int l,int r,int l1,int r1){
if(r<l1||r1<l)return ;
if(l1<=l&&r<=r1)return tr[a].v;
int mid=(l+r)>>;
return (wen(a<<,l,mid,l1,r1)+wen(a<<|,mid+,r,l1,r1))%p;
}
int main(){
n=read(),m=read(),p=read(),c=read();
prime();
for(int i=;i<=n;i++)b[i]=read();
build(,,n);
for(int i=;i<=m;i++){
int op=read(),l=read(),r=read();
if(!op)gai(,,n,l,r);
else printf("%lld\n",wen(,,n,l,r));
}
return ;
}
+++++++++++++++++++++++++++++++++++++++++++
+本文作者:luyouqi233。 +
+欢迎访问我的博客:http://www.cnblogs.com/luyouqi233/+
+++++++++++++++++++++++++++++++++++++++++++
BZOJ4869:[SHOI2017]相逢是问候——题解的更多相关文章
- BZOJ4869 [Shoi2017]相逢是问候 【扩展欧拉定理 + 线段树】
题目链接 BZOJ4869 题解 这题调得我怀疑人生,,结果就是因为某些地方\(sb\)地忘了取模 前置题目:BZOJ3884 扩展欧拉定理: \[c^a \equiv c^{a \mod \varp ...
- Bzoj4869: [Shoi2017]相逢是问候
题面 传送门 Sol 摆定理 \[ a^b\equiv \begin{cases} a^{b\%\phi(p)}~~~~~~~~~~~gcd(a,p)=1\\ a^b~~~~~~~~~~~~~~~~~ ...
- bzoj4869: [Shoi2017]相逢是问候(欧拉函数+线段树)
这题是六省联考的...据说数据还出了点锅,心疼六省选手QAQ 首先要知道扩展欧拉定理... 可以发现每次区间操作都会使模数进行一次phi操作,而一个数最多取logp次phi就会变成1,这时后面的指数就 ...
- 【BZOJ4869】相逢是问候(线段树,欧拉定理)
[BZOJ4869]相逢是问候(线段树,欧拉定理) 题面 BZOJ 题解 根据欧拉定理递归计算(类似上帝与集合的正确用法) 所以我们可以用线段树维护区间最少的被更新的多少次 如果超过了\(\varph ...
- BZOJ:4869: [Shoi2017]相逢是问候
4869: [Shoi2017]相逢是问候 先说点正经的…… 显然做了有限次(我只知道是有限次,而且不会大,别人说是log次?)修改以后会达到不动点,即以后怎么修改都不变了. 然后就随便做了.(3个l ...
- bzoj 4869: [Shoi2017]相逢是问候 [扩展欧拉定理 线段树]
4869: [Shoi2017]相逢是问候 题意:一个序列,支持区间\(a_i \leftarrow c^{a_i}\),区间求和.在模p意义下. 类似于开根操作,每次取phi在log次后就不变了. ...
- 【BZOJ4869】相逢是问候 [线段树][欧拉定理]
相逢是问候 Time Limit: 40 Sec Memory Limit: 512 MB[Submit][Status][Discuss] Description Informatikverbin ...
- 【bzoj4869】[Shoi2017]相逢是问候 扩展欧拉定理+并查集+树状数组
题目描述 Informatik verbindet dich und mich. 信息将你我连结. B君希望以维护一个长度为n的数组,这个数组的下标为从1到n的正整数.一共有m个操作,可以分为两种:0 ...
- 【bzoj4869】[Shoi2017]相逢是问候 线段树+扩展欧拉定理
Description Informatikverbindetdichundmich. 信息将你我连结.B君希望以维护一个长度为n的数组,这个数组的下标为从1到n的正整数.一共有m个操作,可以 分为两 ...
随机推荐
- yarn 原理
产生背景 直接源于MRv1在几个方面的缺陷 扩展性受限(NameNode,JobTracker设计为单一节点,内存容量有限) 单点故障 难以支持MR之外的计算 slot数目无法动态修改,Map slo ...
- FCL中你不得不知的几种委托
FCL中丰富的类库信息极大的方便了我们的编码,很多我们日常经常用到的类型,FCL中已经帮我们定义好,下面要介绍的就是FCL中定义好的几种委托类型,直接使用它们不仅能提高我们的编码效率,而且还能让我们的 ...
- Ubuntu16.04比较好的一系列软件安装介绍
https://blog.csdn.net/Gerald_Jones/article/details/80784976
- 怎样安装Android Studio
在浏览器地址栏输入 http://www.android-studio.org/ 打开Android Studio中文社区, 下载安装包: 这里需要注意的是SDK的目录, 我没有选择默认的目录, 而是 ...
- jmeter3.0 java请求
1.java请求说明 需要压测某些java方法或一些请求需要通过编写代码实现 1.1.依赖jar包: jmeter下/lib/ext中的ApacheJMeter_java.jar(必须).Apache ...
- VMware SDK使用指南
刚开始用VMware官方推荐的SDK,真的是又臭又长,代码结构不清晰,易读性差.后来VMware的同学给推荐了一款开源的SDK,一上手感觉工作效率提高了100倍!推荐大家使用~. 该SDK对VMwar ...
- HDU - 6409:没有兄弟的舞会(数学+思维)
链接:HDU - 6409:没有兄弟的舞会 题意: 题解: 求出最大的 l[i] 的最大值 L 和 r[i] 的最大值 R,那么 h 一定在 [L, R] 中.枚举每一个最大值,那么每一个区间的对于答 ...
- Python实现个性化推荐二
基于内容的推荐引擎是怎么工作的 基于内容的推荐系统,正如你的朋友和同事预期的那样,会考虑商品的实际属性,比如商品描述,商品名,价格等等.如果你以前从没接触过推荐系统,然后现在有人拿枪指着你的头,强迫你 ...
- OJ错误命令解释
①Presentation Error (PE) : 虽然您的程序貌似输出了正确的结果,但是这个结果的格式有点问题. 请检查程序的输出是否多了或者少了空格(' ').制表符('\t')或者换行符('\ ...
- POJ 1995 (快速幂)
这道题普通做法会发生溢出且会超时,应当用快速幂来求解. 快速幂讲解 #include <cstdio> #include <cmath> using namespace std ...
