题目

给一个长度为\(n\)的数组\(a\),\(q\)次询问

支持区间乘\(x\)以及求\(\varphi(\prod_{i=l}^ra_i)\)

\(n\leq 4*10^5,q\leq 2*10^5,x\leq 300\)


分析



之前我做过一道这个题目

方法就是建109棵线段树维护每个位置是否能被某个质数整除,以及一个维护区间乘的线段树

然后我把这道题改一改交到本题上结果发现MLE了,我还没发现这两题的区别,对于之前的这道题,还要记录每个位置质数的个数

但是对于这道题,实际上只需要维护一个二进制状态表示质因数分解的情况,然后取出这个二进制状态计算欧拉函数

时间复杂度\(O(qlogn)\)


代码

#include <cstdio>
#include <cctype>
#define rr register
using namespace std;
typedef long long lll; const int N=400011,mod=1000000007; int n,m,a[N],inv[N];
const int prime[62]={2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53,59,61,67,71,73,79,83,89,97,101,103,107,109,113,127,131,137,139,149,151,157,163,167,173,179,181,191,193,197,199,211,223,227,229,233,239,241,251,257,263,269,271,277,281,283,293};
inline signed iut(){
rr int ans=0; rr char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();
return ans;
}
inline signed ksm(int x,int y){
rr int ans=1;
for (;y;y>>=1,x=1ll*x*x%mod)
if (y&1) ans=1ll*ans*x%mod;
return ans;
}
inline void print(int ans){
if (ans>9) print(ans/10);
putchar(ans%10+48);
}
struct rec{
lll w1,w2;
rec operator *(const rec &t)const{
return (rec){w1*t.w1%mod,w2|t.w2};
}
};
struct segt{
lll lazy[N<<2],Lazy[N<<2]; rec w[N<<2];
inline void build(int k,int l,int r){
lazy[k]=1,Lazy[k]=0;
if (l==r){
w[k].w1=a[l];
for (rr int i=0;i<62;++i)
if (a[l]%prime[i]==0)
w[k].w2|=1ll<<i;
return;
}
rr int mid=(l+r)>>1;
build(k<<1,l,mid);
build(k<<1|1,mid+1,r);
w[k]=w[k<<1]*w[k<<1|1];
}
inline void spread(int k,int len,int z,lll Z){
lazy[k]=lazy[k]*z%mod,Lazy[k]|=Z;
w[k]=(rec){w[k].w1*ksm(z,len)%mod,w[k].w2|=Z};
}
inline void pushdown(int k,int l,int r,int mid){
if (!Lazy[k]) return;
rr int len1=mid-l+1,len2=r-mid;
spread(k<<1,len1,lazy[k],Lazy[k]);
spread(k<<1|1,len2,lazy[k],Lazy[k]);
lazy[k]=1,Lazy[k]=0;
}
inline rec query(int k,int l,int r,int x,int y){
if (l==x&&r==y) return w[k];
rr int mid=(l+r)>>1; pushdown(k,l,r,mid);
if (y<=mid) return query(k<<1,l,mid,x,y);
else if (x>mid) return query(k<<1|1,mid+1,r,x,y);
else return query(k<<1,l,mid,x,mid)*query(k<<1|1,mid+1,r,mid+1,y);
}
inline void update(int k,int l,int r,int x,int y,int z,lll Z){
if (l==x&&r==y){
spread(k,r-l+1,z,Z);
return;
}
rr int mid=(l+r)>>1; pushdown(k,l,r,mid);
if (y<=mid) update(k<<1,l,mid,x,y,z,Z);
else if (x>mid) update(k<<1|1,mid+1,r,x,y,z,Z);
else update(k<<1,l,mid,x,mid,z,Z),update(k<<1|1,mid+1,r,mid+1,y,z,Z);
w[k]=w[k<<1]*w[k<<1|1];
}
}tre;
signed main(){
n=iut(),m=iut();
for (rr int i=0;i<62;++i) inv[i]=1ll*(prime[i]-1)*ksm(prime[i],mod-2)%mod;
for (rr int i=1;i<=n;++i) a[i]=iut();
tre.build(1,1,n);
for (rr char c;m;--m){
c=getchar();
while (c!='T'&&c!='M') c=getchar();
rr int l=iut(),r=iut();
if (c=='T'){
rr rec ANS=tre.query(1,1,n,l,r);
for (rr int i=0;i<62;++i)
if ((ANS.w2>>i)&1){
ANS.w1=1ll*ANS.w1*inv[i]%mod;
}
print(ANS.w1),putchar(10);
}
else{
rr int x=iut(); rr lll T=0;
if (x==1) continue;
for (rr int i=0;i<62;++i)
if (x%prime[i]==0) T|=1ll<<i;
tre.update(1,1,n,l,r,x,T);
}
}
return 0;
}

#线段树,欧拉函数#CF1114F Please, another Queries on Array?的更多相关文章

  1. 线段树+欧拉函数——cf1114F

    调了半天,写线段树老是写炸 /* 两个操作 1.区间乘法 2.区间乘积询问欧拉函数 欧拉函数计算公式 phi(mul(ai))=mul(ai) * (p1-1)/p1 * (p2-1)/p2 * .. ...

  2. BZOJ 3813--奇数国(线段树&欧拉函数&乘法逆元&状态压缩)

    3813: 奇数国 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 755  Solved: 432[Submit][Status][Discuss] ...

  3. 【bzoj3813】: 奇数国 数论-线段树-欧拉函数

    [bzoj3813]: 奇数国 题意:给定一个序列,每个元素可以分解为最小的60个素数的形式.(x=p1^k1*p2^k2*......p60^k60)(p1=2,p2=3,…,p60=281) 支持 ...

  4. [bzoj3813] 奇数国 [线段树+欧拉函数]

    题面 传送门 思路 这题目是真的难读......阅读理解题啊...... 但是理解了以后就发现,题目等价于: 给你一个区间,支持单点修改,以及查询一段区间的乘积的欧拉函数值,这个答案对19961993 ...

  5. Please, another Queries on Array?(Codeforces Round #538 (Div. 2)F+线段树+欧拉函数+bitset)

    题目链接 传送门 题面 思路 设\(x=\prod\limits_{i=l}^{r}a_i\)=\(\prod\limits_{i=1}^{n}p_i^{c_i}\) 由欧拉函数是积性函数得: \[ ...

  6. Please, another Queries on Array? CodeForces - 1114F (线段树,欧拉函数)

    这题刚开始看成求区间$\phi$和了........先说一下区间和的做法吧...... 就是说将题目的操作2改为求$(\sum\limits_{i=l}^{r}\phi(a[i]))\%P$ 首先要知 ...

  7. BZOJ4869 六省联考2017相逢是问候(线段树+欧拉函数)

    由扩展欧拉定理,a^(a^(a^(……^x)))%p中x作为指数的模数应该是φ(φ(φ(φ(……p)))),而p取log次φ就会变为1,也即每个位置一旦被修改一定次数后就会变为定值.线段树维护区间剩余 ...

  8. 【BZOJ3813】奇数国 线段树+欧拉函数

    [BZOJ3813]奇数国 Description 给定一个序列,每次改变一个位置的数,或是询问一段区间的数的乘积的phi值.每个数都可以表示成前60个质数的若干次方的乘积. Sample Input ...

  9. BZOJ 4026: dC Loves Number Theory 可持久化线段树 + 欧拉函数 + 数学

    Code: #include <bits/stdc++.h> #define ll long long #define maxn 50207 #define setIO(s) freope ...

  10. BZOJ_4026_dC Loves Number Theory _主席树+欧拉函数

    BZOJ_4026_dC Loves Number Theory _主席树+欧拉函数 Description  dC 在秒了BZOJ 上所有的数论题后,感觉萌萌哒,想出了这么一道水题,来拯救日益枯 竭 ...

随机推荐

  1. Vue3学习(十八) - TreeSelect 树选择

    写在前面 本以为可以在家学习一天,结果家里来了客人拜年,就没学习上,有点小遗憾吧. 昨天完成从分类管理的前后端代码复制出文档管理的前后端代码,遗留问题是只能选择一级父分类.值得说的是,昨晚的遗留的问题 ...

  2. 项目实战:Qt+Android模拟操作器(模拟操作app,打开,点击,输入,获取验证码等等)

    若该文为原创文章,转载请注明原文出处本文章博客地址:https://blog.csdn.net/qq21497936/article/details/109313803各位读者,知识无穷而人力有穷,要 ...

  3. ABP开发需要用到的命令

    0.命令行在哪里执行? 在Visual Studio的"解决方案资源管理器"的解决方案或者项目上点鼠标右键,选择"在终端中打开". 1.安装abp的命令行 官网 ...

  4. 如何使用 perf 分析 splice 中 pipe 的容量变化

    如何使用 perf 分析 splice 中 pipe 的容量变化 这个文章为了填上一篇文章的坑的,跟踪内核函数本来是准备使用 ebpf 的,但是涉及到了低内核版本,只能使用 kprobe 了. 恰好, ...

  5. 主流开源分布式图计算框架 Benchmark

    本文由美团 NLP 团队高辰.赵登昌撰写,首发于 Nebula Graph Community 公众号 前言 随着近年来数据的爆炸式增长,如何高效地分析处理数据,在业界一直备受关注.现实世界中的数据往 ...

  6. 利用微软官方API实现Office文档的在线预览功能

    随着互联网时代的飞速发展,越来越多的工作开始依赖于云端服务,我们的办公方式也逐渐发生了翻天覆地的变化.在这种背景下,急需一种无需本地安装Office软件,就能快速查看和共享Word.PowerPoin ...

  7. OpenCV开发笔记(七十六):相机标定(一):识别棋盘并绘制角点

    前言   知道图像畸变矫映射的原理之后,那么如何得到相机的内参是矫正的第一步,内参决定了内参矩阵(中心点.焦距等),用内参矩阵才能计算出投影矩阵,从而将原本畸变的图像矫正为平面投影图像.  本篇描述了 ...

  8. Elasticsearch 如何保证写入过程中不丢失数据的

    丢失数据的本质 在本文开始前,首先明白一个点,平时我们说的组件数据不丢失究竟是在指什么,如果你往ES写入数据,ES返回给你写入错误,这个不算数据丢失.如果你往ES写入数据,ES返回给你成功,但是后续因 ...

  9. 单词本z launch = to throw 相关词根 lanc ject jac jet bol bl bal mit miss cast

    launch = to throw 词根 launc = lanc = to throw 表示 to throw 的词根有三组 拉丁语的lanc/launc,sip 拉丁语的ject, jac(jet ...

  10. 微型MPU6050模块及串口/BLE透传的实物展示

    前言   随着时间的积累,项目的沉淀.把很多做过的产品标准化掉,形成可以立即拿出来使用的产品.是非常重要的一个环节.最近,把基于MPU6050的模块微型化并使用串口透传这个产品梳理了一下.形成了标准化 ...