BZOJ 4869: [Shoi2017]相逢是问候

题目大意:

维护一个序列,支持将区间内的数x变为c^x,区间求和。

题解:
扩展欧拉定理

a^b=a^(b%phi(p)+phi(p)) ( mod p )

然后因为底数是确定的,所以一个数最多运行log(x)次就不会变了,因为phi(p)执行log(x)次后变为1

线段树暴力即可

代码:

#include<cstdio>
#include<algorithm>
using namespace std;
int n,mod,c,m,p,cnt,a[1000005],phi[1000005],tree[1000005],minn[1000005];
int get_phi(int x){
	int ans=x;
	for (int i=2; i*i<=x; i++)
		if (x%i==0){
			ans=ans/i*(i-1);
			while (x%i==0) x/=i;
		}
	if (x!=1) ans=ans/x*(x-1);
	return ans;
}
void build(int t,int l,int r){
	if (l==r){
		tree[t]=a[l];
		return;
	}
	int mid=(l+r)>>1;
	build(t<<1,l,mid);
	build(t<<1|1,mid+1,r);
	tree[t]=(tree[t<<1]+tree[t<<1|1])%mod;
}
int pow(int a,int b,int mod){
	int ans=1;
	while (b){
		if (b&1) ans=1ll*ans*a%mod;
		a=1ll*a*a%mod;
		b=b>>1;
	}
	return ans;
}
int query(int a,int times){
	for (int i=times; i>=1; i--){
		if (a>=phi[i]) a=a%phi[i]+phi[i];
		a=pow(c,a,phi[i-1]);
		if (!a) a=phi[i-1];
	}
	return a;
}
void change(int t,int l,int r,int x,int y){
	if (l>y || r<x) return;
	if (minn[t]>=cnt) return;
	if (l==r){
		minn[t]++;
		tree[t]=query(a[l],minn[t]);
		return;
	}
	int mid=(l+r)>>1;
	change(t<<1,l,mid,x,y);
	change(t<<1|1,mid+1,r,x,y);
	tree[t]=(tree[t<<1]+tree[t<<1|1])%mod;
	minn[t]=min(minn[t<<1],minn[t<<1|1]);
}
int query(int t,int l,int r,int x,int y){
	if (l>y || r<x) return 0;
	if (l>=x && r<=y) return tree[t];
	int mid=(l+r)>>1;
	return (query(t<<1,l,mid,x,y)+query(t<<1|1,mid+1,r,x,y))%mod;
}
int main(){
	scanf("%d%d%d%d",&n,&m,&p,&c);
	for (int i=1; i<=n; i++)
		scanf("%d",&a[i]);
	mod=phi[0]=p;
	while (p!=1){
		phi[++cnt]=get_phi(p);
		p=phi[cnt];
	}
	phi[++cnt]=1;
	build(1,1,n);
	while (m--){
		int cas,l,r;
		scanf("%d%d%d",&cas,&l,&r);
		if (cas==0) change(1,1,n,l,r);
		else printf("%d\n",query(1,1,n,l,r));
	}
	return 0;
}

  

posted @ 2018-04-23 19:33  ~Silent  阅读(261)  评论(0编辑  收藏  举报
Live2D