codeforces CF920F SUM and REPLACE 线段树 线性筛约数
F. SUM and REPLACE
Let D(x) be the number of positive divisors of a positive integer x .
For example, D(2) = 2 ( 2 is divisible by 1 and 2 ), D(6) = 4 ( 6 is divisible by 1,2,3 and 6 ).
You are given an array a of n integers. You have to process two types of queries:
1.REPLACElr —— for every i∈[l,r] replace ai with D(ai) ;
2.SUMlr —— calculate ∑ri=lai .
Print the answer for each SUM query.
Input
The first line contains two integers n and m(1 ≤ n, m ≤ 3·105)
— the number of elements in the array and the number of queries to process, respectively.
The second line contains n integers a1,a2,...,an(1 ≤ ai ≤ 106) — the elements of the array.
Then m lines follow, each containing 3 integers ti,li,ri denoting i -th query.
If ti = 1 , then i -th query is REPLACEliri , otherwise it's SUMliri(1 ≤ ti ≤ 2,1 ≤ li ≤ ri ≤ n) .
There is at least one SUM query.
Output
For each SUM query print the answer to it.
Examples
input
7 6
6 4 1 10 3 2 4
2 1 7
2 4 5
1 3 5
2 4 4
1 5 7
2 1 7
output
30
13
4
22
题目大意
-
给你 n 个数,进行 m 次操作,分别是将区间 [l,r] 内的所有数替换成自己的因子数 和 对区间 [l,r] 进行求和。
-
数据范围: 1≤n,m≤3×105
思路
- 首先可以用线段树来维护序列,每次暴力修改。
因为 D(1)=1,D(2)=2,因此当一个区间的最大值小于2时就不需要修改。
- 又因为每次修改只会让值减少,而且每个数最多会被修改 O(logn) 次,因此能够保证复杂度
代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define MAXM 1000005
#define N 300005
#define int long long
int n,m,M;
int d[MAXM],c[MAXM],p[MAXM];
bool vis[MAXM];
void pre(){
d[1]=1;
for(int i=2;i<=M;++i){
if(!vis[i]){ p[++p[0]]=i; vis[i]=1; d[i]=2; c[i]=1; }
for(int j=1;j<=p[0]&&1ll*i*p[j]<=M;++j){
vis[i*p[j]]=1;
if(i%p[j]==0){
d[i*p[j]]=d[i]/(c[i]+1)*(c[i]+2);
c[i*p[j]]=c[i]+1;
break;
}
d[i*p[j]]=d[i]*2;
c[i*p[j]]=1;
}
}
}
int sum[N<<2];
bool flag[N<<2];
void build(int o,int l,int r){
if(l==r){
scanf("%lld",&sum[o]);
if(sum[o]==1||sum[o]==2) flag[o]=1;
M=max(M,sum[o]);
return;
}
int mid=l+r>>1;
build(o<<1,l,mid); build(o<<1|1,mid+1,r);
sum[o]=sum[o<<1]+sum[o<<1|1];
flag[o]=flag[o<<1]&flag[o<<1|1];
}
void updata(int o,int l,int r,int L,int R){
if(flag[o]) return;
if(l==r&&L<=l&&l<=R){
sum[o]=d[sum[o]];
if(sum[o]==1||sum[o]==2) flag[o]=1;
return;
}
int mid=l+r>>1;
if(L>mid) updata(o<<1|1,mid+1,r,L,R);
else if(R<=mid) updata(o<<1,l,mid,L,R);
else {
updata(o<<1,l,mid,L,R);
updata(o<<1|1,mid+1,r,L,R);
}
sum[o]=sum[o<<1]+sum[o<<1|1];
flag[o]=flag[o<<1]&flag[o<<1|1];
}
int query(int o,int l,int r,int L,int R){
if(L<=l&&r<=R) return sum[o];
int mid=l+r>>1;
if(L>mid) return query(o<<1|1,mid+1,r,L,R);
else if(R<=mid) return query(o<<1,l,mid,L,R);
else return query(o<<1,l,mid,L,R)+query(o<<1|1,mid+1,r,L,R);
}
signed main(){
scanf("%lld %lld",&n,&m);
build(1,1,n);
pre();
while(m--){
int opt,x,y;
scanf("%lld %lld %lld",&opt,&x,&y);
if(opt==1) updata(1,1,n,x,y);
else printf("%lld\n",query(1,1,n,x,y));
}
return 0;
}
/*
# 42607112
When 2018-09-07 11:38:57
Who PotremZ
Problem F - SUM and REPLACE
Lang GNU C++11
Verdict Accepted
Time 467 ms
Memory 35000 KB
*/
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步