HYSBZ - 3813 奇数国 欧拉函数+树状数组(线段树)

HYSBZ - 3813奇数国

  中文题,巨苟题,巨无敌苟!!首先是关于不相冲数,也就是互质数的处理,欧拉函数是可以求出互质数,但是这里的product非常大,最小都2100000,这是不可能实现的。所以我们要求互质数的话,得用到所有金额都用60个素数表示的这个条件。也就是x=p1a1xp2a2x...p60a60表示,pi是第i个素数,ai是对应的指数,这就变成了互质素求欧拉函数,可以先了解一下欧拉函数,引用一下境外大佬的博客欧拉函数的讲解我们需要用到这一条

p为质数

1. phi(p)=p-1   因为质数p除了1以外的因数只有p,故1至p的整数只有p与p不互质 

2. 如果i mod p = 0, 那么 phi(i * p)=phi(i) * p

3.若i mod p ≠0,  那么 phi( i * p )=phi(i) * ( p-1 ) 

  首先,如果我们要求phi(p1a1)的话,p1是质数,然后p1a1=p1a1-1*p1,而p1a1-1 mod p1=0,所以phi(p1a1)=phi(p1a1-1)*p1,又phi(p1a1-1)=phi(p1a1-2)*p1,一直到,phi(p1)=p1-1,所以phi(p1a1)=p1a1-1*(p1-1),

  然后求phi(p1a1*p2),p1a1 mod p2≠0,所以phi(p1a1*p2)=phi(p1a1)*(p2-1),再算phi(p1a1*p22),p1a1*p22 mod p2=0,所以phi(p1a1*p22)=phi(p1a1*p2)*p2,所以这样推下去的话,phi(p1a1*p2a2)=p1a1-1*(p1-1)*p2a2-1*(p2-1)

  我们可以得到phi(p1a1xp2a2x...p60a60)=p1a1-1*(p1-1)*p2a2-1*(p2-1)*...*p60a60-1*(p60-1)。

  因为它说这个国家的加法就是我们的乘法,所以我们可以用树状数组或者线段树来维护1~100000里存储的每个数对应的素数的次方,但第二苟的点来了,如果用线段树还好说,只要处理好懒标记就应该可以了,但是用树状数组,单点更新那里一不小心就会超时,超时了好几发后,我看学长的代码原来他们的树状数组也是9492ms险过的,先看AC代码,更新处是使用学长的处理。

 1 #include<cstdio>
 2 #define lowb(x) x&(-x)
 3 #define ll long long
 4 const int  N=100000;
 5 const ll mod=19961993;
 6 int a[N+18][66]={0},b[N],prime[66]={
 7 0,2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53,59,61,67,
 8 71,73,79,83,89,97,101,103,107,109,113,127,131,137,139,
 9 149,151,157,163,167,173,179,181,191,193,197,199,211,223,
10 227,229,233,239,241,251,257,263,269,271,277,281};//我自己把素数打出来了 
11 void updata(int x,int y,int z)
12 {
13     while(x<=N)
14     {
15         a[x][y]+=z;
16         x+=lowb(x);
17     }
18 }
19 int geta(int x,int y)
20 {
21     int ans=0;
22     while(x)
23     {
24         ans+=a[x][y];
25         x-=lowb(x);
26     }
27     return ans;
28 }
29 void modify(int x,int y,int flag)
30 {
31     for(int i=1;i<=60;i++)//把价值转化成p1^a1+p2^a2+p3^a3+...+p60^a60的形式,然后存储指数ai 
32         if(y%prime[i]==0)
33         {
34             int z=0;
35             while(y%prime[i]==0)
36             {
37                 z++;
38                 y/=prime[i];
39             }
40             updata(x,i,flag*z);
41         }
42 }
43 ll pow(ll a,int b)
44 {
45     ll ans=1;
46     a%=mod;
47     while(b)
48     {
49         if(b&1)
50             ans=(ans*a)%mod;
51         a=(a*a)%mod;
52         b>>=1;
53     }
54     return ans;
55 }
56 int main()
57 {
58     int n,op,x,y,z;
59     scanf("%d",&n);
60     for(int i=1;i<=N;i++)
61     {
62         b[i]=3;
63         updata(i,2,1);//3是第二个素数,一开始每个都有3块钱 
64     }
65     while(n--)
66     {
67         scanf("%d%d%d",&op,&x,&y);
68         if(op)
69         {
70             modify(x,b[x],-1);//先减去原来的数 
71             modify(x,y,1);//再加上要修改的数 
72             b[x]=y;
73         }
74         else
75         {
76             ll ans=1;
77             for(int i=1;i<=60;i++)
78             {
79                 z=geta(y,i)-geta(x-1,i);
80                 if(z)
81                     ans=ans*pow(1ll*prime[i],z-1)*(prime[i]-1)%mod;
82             }
83             printf("%lld\n",ans);
84         }
85     }
86     return 0;
87 }
学长好腻害哦

  学长的处理就是,保存好每一个原来的存款,然后每次更新时,先减去旧的存款然后再加上新的存款,而我的处理的话是

 1 for(int i=1;i<=60;i++)
 2 {
 3     z=0;
 4     while(y%prime[i]==0)
 5     {
 6         z++;
 7         y/=prime[i];    
 8     }
 9     z=z-a[x][i];//原来新的指数和对旧的指数差值,正的说明要加上,负的说明要减去,0不需要更新
10     if(z!=0)
11       updata(x,i,z);    
12 }

  我觉得我这样每个i只进行了一次更新,应该比学长的更快,然而T了,我觉得原因应该是在于如果z不是0的话,那么我的就是60遍每次都要从x节点更新到100000为止,但学长的看上去是跑了两边,但是if(y%prime[i]==0)这一句过滤掉很多没必要的更新,所以还是学长nb啊。。。

  至于线段树的做法,先留个坑,毕竟女同志能顶半边天。

posted @ 2019-03-18 22:59  新之守护者  阅读(170)  评论(0编辑  收藏  举报