[BZOJ 4809] 相逢是问候
Link:
Solution:
以前没见过的套路题……
1、使用EXT欧拉定理降幂的套路:
$a^{x}=a^{xmod\phi(P)+\phi(P)} mod P$,且$x\ge P$
这样对于$c^{c^{c^x}}modP$就能递推/递归得套用上述定理计算,每层模数多套一层$\phi$即可
注意每次在快速幂时要判断当前指数是否大于当前模数才能用EXT!
2、能证明一个数最多求$log$次$\phi$就会变成1
这样在$log$次内暴力更新,否则不管,就能保证$O(n*log^3)$
3、复杂度中的3个$log$分别是:
更新$log$次,每次更新迭代$log$层,每层要算一次快速幂
明显只能优化快速幂。由于底不变,模数只有$log$种,想到分数的前后两部分预处理!
分块预处理出$[1,(1<<16)]$和$[1*(1<<16),(1<<16)*(1<<16)]$的答案以及与模数的大小关系
这样每次拆出指数的前16位和后16位$O(1)$计算答案和大小关系就能做到$O(n*log^2)$
4、听说原题数据锅了……
虽然$\phi(2)$和$\phi(1)$都为1,但要更新到$\phi(1)$!
否则在最顶层指数为0时最终会迭代出$cmod2$而非$cmod1$,不一定为0!
Code:
#include <bits/stdc++.h> using namespace std;
#define X first
#define Y second
#define pb push_back
typedef double db;
typedef long long ll;
typedef pair<int,int> P;
const int MAXN=1e5+;
ll pre[][<<][];
bool f[][<<][];
int n,m,p,c,dat[MAXN],phi[],cnt; int getphi(int x)
{
int ret=x;
for(int i=;i*i<=x;i++)
if(x%i==)
{
ret=ret/i*(i-);
while(x%i==) x/=i;
}
if(x!=) ret=ret/x*(x-);
return ret;
}
ll quick_pow(ll a,ll b,ll MOD,bool &f)
{
ll ret=;
for(;b;b>>=,a=a*a%MOD)
{
//判断是否大于某个数两个地方都要判!
if(b&) f|=(ret*a>=MOD),ret=ret*a%MOD;
f|=(a*a>=MOD&&b!=);
}
return ret;
}
int QP(int x,int num,bool &flag)
{
int a=x&((<<)-),b=x>>;
ll ret=1ll*pre[num][a][]*pre[num][b][];
flag=f[num][a][]|f[num][b][]|(ret>=phi[num]);
return ret%phi[num];
}
int cal(int x,int st)
{
int ret=x;
if(ret>=phi[st])
ret=ret%phi[st]+phi[st];
while(st--)
{
bool f=;
ret=QP(ret,st,f);
//注意特判,仅在指数>模数时可使用EXT欧拉定理
if(f&&st) ret+=phi[st];
}
return ret%p;
}
void PRE()
{
//分块预处理
for(int i=;i<=cnt;i++)
{
pre[i][][]=pre[i][][]=;
if(phi[i]==) f[i][][]=f[i][][]=;
pre[i][][]=quick_pow(c,,phi[i],f[i][][]);
int tmp=pre[i][][]=quick_pow(c,<<,phi[i],f[i][][]); for(int j=;j<<<;j++)
{
f[i][j][]=f[i][j-][]|(pre[i][j-][]*c>=phi[i]);
pre[i][j][]=pre[i][j-][]*c%phi[i];
f[i][j][]=f[i][j-][]|(pre[i][j-][]*tmp>=phi[i]);
pre[i][j][]=pre[i][j-][]*tmp%phi[i];
}
}
} namespace SegmentTree
{
#define mid ((l+r)>>1)
#define ls k<<1
#define rs k<<1|1
#define lc ls,l,mid
#define rc rs,mid+1,r int sum[MAXN<<],tag[MAXN<<];
void pushup(int k)
{
tag[k]=min(tag[ls],tag[rs]);
sum[k]=(sum[ls]+sum[rs])%p;
}
void build(int k,int l,int r)
{
if(l==r)
{sum[k]=dat[l]%p;tag[k]=;return;}
build(lc);build(rc);pushup(k);
}
void modify(int a,int b,int k,int l,int r)
{
if(tag[k]>=cnt) return;
if(l==r)
{sum[k]=cal(dat[l],++tag[k]);return;}
if(a<=mid) modify(a,b,lc);
if(b>mid) modify(a,b,rc);
pushup(k);
}
int query(int a,int b,int k,int l,int r)
{
if(a<=l&&r<=b) return sum[k];
int ret=;
if(a<=mid) (ret+=query(a,b,lc))%=p;
if(b>mid) (ret+=query(a,b,rc))%=p;
return ret;
}
}
using namespace SegmentTree; int main()
{
scanf("%d%d%d%d",&n,&m,&p,&c);
for(int i=;i<=n;i++)
scanf("%d",&dat[i]);
phi[]=p;
while(phi[cnt]!=)
cnt++,phi[cnt]=getphi(phi[cnt-]);
//要迭代到phi(1)=1,而不能仅迭代到phi(2)=1
//否则在x=0时最后一层会出现c%2
phi[++cnt]=;PRE(); build(,,n);
while(m--)
{
int op,l,r;
scanf("%d%d%d",&op,&l,&r);
if(!op) modify(l,r,,,n);
else printf("%d\n",query(l,r,,,n));
}
return ;
}
[BZOJ 4809] 相逢是问候的更多相关文章
- BZOJ:4869: [Shoi2017]相逢是问候
4869: [Shoi2017]相逢是问候 先说点正经的…… 显然做了有限次(我只知道是有限次,而且不会大,别人说是log次?)修改以后会达到不动点,即以后怎么修改都不变了. 然后就随便做了.(3个l ...
- bzoj 4869: [Shoi2017]相逢是问候 [扩展欧拉定理 线段树]
4869: [Shoi2017]相逢是问候 题意:一个序列,支持区间\(a_i \leftarrow c^{a_i}\),区间求和.在模p意义下. 类似于开根操作,每次取phi在log次后就不变了. ...
- 【BZOJ4869】相逢是问候(线段树,欧拉定理)
[BZOJ4869]相逢是问候(线段树,欧拉定理) 题面 BZOJ 题解 根据欧拉定理递归计算(类似上帝与集合的正确用法) 所以我们可以用线段树维护区间最少的被更新的多少次 如果超过了\(\varph ...
- [BZOJ4869][六省联考2017]相逢是问候(线段树+扩展欧拉定理)
4869: [Shoi2017]相逢是问候 Time Limit: 40 Sec Memory Limit: 512 MBSubmit: 1313 Solved: 471[Submit][Stat ...
- 【BZOJ4869】相逢是问候 [线段树][欧拉定理]
相逢是问候 Time Limit: 40 Sec Memory Limit: 512 MB[Submit][Status][Discuss] Description Informatikverbin ...
- 洛谷 P3747 [六省联考2017]相逢是问候 解题报告
P3747 [六省联考2017]相逢是问候 题目描述 \(\text {Informatik verbindet dich und mich.}\) 信息将你我连结. \(B\) 君希望以维护一个长度 ...
- 2017 [六省联考] T2 相逢是问候
4869: [Shoi2017]相逢是问候 Time Limit: 40 Sec Memory Limit: 512 MBSubmit: 1205 Solved: 409[Submit][Stat ...
- 洛谷P3747 [六省联考2017]相逢是问候
传送门 题解 扩展欧拉定理. 线段树维护,已经全改到底了的节点就不管,不然暴力修改下去. //Achen #include<algorithm> #include<iostream& ...
- bzoj千题计划271:bzoj4869: [六省联考2017]相逢是问候
http://www.lydsy.com/JudgeOnline/problem.php?id=4869 欧拉降幂+线段树,每个数最多降log次,模数就会降为1 #include<cmath&g ...
随机推荐
- js数据类型隐式转换问题
js数据类型隐式转换 ![] == false //true 空数组和基本类型转换,会先[].toString() 再继续比较 ![] == [] //true ![] //false [] == [ ...
- 【SVN】centos环境下搭建SVN服务器
1.安装SVN,有些linux发行版自带SVN,可以用下面方法检测是否安装SVN. svn --version 如果 Subversion 客户端没有安装,命令将报告svn命令找不到的错误. 我们可以 ...
- mvn打war包以及解压包的方法
有时候我们需要查看打成war包之后的目录,如果是maven项目我们可以直接用maven打包. 1.maven打包: 第一种: mvn package 如果不行先 mvn clean一下 第二种:(掌握 ...
- 你需要知道的12个Git高级命令【转】
转自:http://www.linuxidc.com/Linux/2016-01/128024.htm 众所周知,Git目前已经是分布式版本控制领域的翘楚,围绕着Git形成了完整的生态圈.学习Git, ...
- oracle11g创建修改删除表
oracle11g创建修改删除表 我的数据库名字: ORCL 密码:123456 1.模式 2.创建表 3.表约束 4.修改表 5.删除表 1.模式 set oracle_sid=OR ...
- Django-自动HTML转义
一.自动HTML转义 从模板生成HTML时,总会有变量包含影响最终HTML的字符风险,例如,考虑这个模板的片段: Hello, {{ name }} 起初,这是一种显示用户名的无害方式,但考虑用户输入 ...
- DenseNet笔记
一.DenseNet的优点 减轻梯度消失问题 加强特征的传递 充分利用特征 减少了参数量 二.网络结构公式 对于每一个DenseBlock中的每一个层, [x0,x1,…,xl-1]表示将0到l-1层 ...
- 三、springcloud之服务调用Feign
一.背景 项目中接口调用: Httpclient Okhttp Httpurlconnection RestTemplate 微服务提供了更简单,方便的Feign 二.Feign简介 Feign是一个 ...
- 一种获取xml文件某个节点内容的shell方法
配置文件 config.xml <xml> <server> <name>srv-01</name> </server> <serve ...
- ls和cd命令详解
ls命令 命令功能: 列出当前目录下或者指定目录下的所有文件和目录,ls是list的缩写. 命令语法: ls [选项][目录名 ] #注:[ ]中的内容为非必选项 命令选项: | 选项 | 含义 ...