zoukankan      html  css  js  c++  java
  • 2017省夏令营Day7

    题解:首先,我们可以得到一个规律:经过2次变换后,a和b的值都分别乘2了,所以只要用快速幂就能过啦,但是,要特判n为0的情况。

    代码如下:

     1 #include<cstdio>
     2 #include<iostream>
     3 #include<cstring>
     4 #define Mod 1000000007
     5 using namespace std;
     6 long long a,b,n,ans1,ans2;
     7 long long power(long long x){
     8     long long ret=1,tmp=2;
     9     while(x){
    10         if(x&1){ret*=tmp; ret%=Mod;}
    11         tmp*=tmp; tmp%=Mod; x/=2;
    12     }
    13     return ret;
    14 }
    15 int main()
    16 {
    17     freopen("apexis.in","r",stdin);
    18     freopen("apexis.out","w",stdout);
    19     scanf("%lld%lld%lld",&a,&b,&n);
    20     if(n==0){
    21         printf("%lld %lld",a,b);
    22         return 0;
    23     }
    24     if(a<b) swap(a,b);
    25     long long tmp=n>>1;
    26     long long t=power(tmp);
    27     if(n&1) ans1=t*(a+b)%Mod,ans2=t*(a-b)%Mod;
    28     else ans1=t*a%Mod,ans2=t*b%Mod;
    29     printf("%lld %lld",ans1,ans2);
    30     return 0;
    31 }

    ------------------------------------------------------------华丽的分割线----------------------------------------------------------------------


    题解:我们记f[i]表示区间[1,i]内素数个数,我们可以用筛法筛出数据范围内的素数并顺便求f数组,然后我们暴力枚举1-maxn中的数即可。

    代码如下:

     1 #include<cstdio>
     2 #include<iostream>
     3 #define Max 200005
     4 using namespace std;
     5 int n,m,ans,cnt,maxn;
     6 int f[Max],num[Max],prime[Max];
     7 void shaifa(){
     8     for(int i=1;i<=Max;i++) f[i]=i;
     9     for(int i=2;i<=Max;i++)
    10         if(!prime[i]){
    11             f[i]--;
    12             for(int j=2*i;j<=Max/2;j+=i){
    13                 f[j]=f[j]*(i-1)/i;
    14                 prime[j]=1;
    15             }
    16         }
    17 }
    18 int main()
    19 {
    20     freopen("quest.in","r",stdin);
    21     freopen("quest.out","w",stdout);
    22     shaifa();
    23     int T; scanf("%d",&T);
    24     while(T--){
    25         ans=cnt=maxn=0;
    26         scanf("%d%d",&n,&m);
    27         for(int i=1;i<=n;i++){
    28             int x; scanf("%d",&x);
    29             num[x]++; maxn=max(maxn,x);
    30         }
    31         for(int i=1;i<=maxn;i++){
    32             cnt=0;
    33             for(int j=i;j<=maxn;j+=i) cnt+=num[j];
    34             if(cnt>=m) ans=max(ans,f[i]);
    35             num[i]=0;
    36         }
    37         printf("%d
    ",ans);
    38     }
    39     return 0;
    40 }

    ------------------------------------------------------------华丽的分割线----------------------------------------------------------------------

    题解(假):首先,在n、m<=100时,我们可以暴力使枚举l到r,然后用矩阵快速幂求出每个点的答案,最后加起来即可,时间复杂度约为为O(NMlogK)。

    代码如下:当做矩阵快速幂的练习吧,最蠢的暴力- -

     1 #include<cstdio>
     2 #include<iostream>
     3 #include<cstring>
     4 #define Max 100001
     5 #define Mod 10007
     6 using namespace std;
     7 int n,m,l,r;
     8 long long K,cnt[Max];
     9 struct mat{long long num[2][2];}a,b,c,ret;
    10 mat mult(mat x,mat y){
    11     memset(c.num,0,sizeof(c.num));
    12     for(int i=0;i<2;i++)
    13         for(int j=0;j<2;j++)
    14             for(int k=0;k<2;k++)
    15                 c.num[i][j]=(c.num[i][j]+x.num[i][k]*y.num[k][j])%Mod;
    16     return c;
    17 }
    18 mat matmod(mat x,long long k){
    19     memset(ret.num,0,sizeof(ret.num));
    20     for(int i=0;i<2;i++) ret.num[i][i]=1;
    21     while(k){
    22         if(k&1) ret=mult(ret,x);
    23         k>>=1; x=mult(x,x);
    24     }
    25     return ret;
    26 }
    27 long long Get(long long k){
    28     memset(a.num,0,sizeof(a.num));
    29     memset(b.num,0,sizeof(b.num));
    30     a.num[0][1]=1; a.num[1][0]=1; a.num[1][1]=1;
    31     b.num[0][0]=1; b.num[1][0]=1;
    32     a=matmod(a,k-1);
    33     a=mult(a,b);
    34     return a.num[0][0];
    35 }
    36 int main()
    37 {
    38     freopen("rabbit.in","r",stdin);
    39     freopen("rabbit.out","w",stdout);
    40     scanf("%d%d",&n,&m);
    41     for(int i=1;i<=n;i++) cnt[i]=1;
    42     for(int i=1;i<=m;i++){
    43         long long ans=0;
    44         scanf("%d%d%lld",&l,&r,&K);
    45         if(K==0){
    46             for(int j=l;j<=r;j++)
    47                 if(cnt[j]!=0) ans=(ans+Get(cnt[j]))%Mod;
    48             printf("%lld
    ",ans);
    49         }
    50         else for(int j=l;j<=r;j++) cnt[j]+=K;
    51     }
    52     return 0;
    53 }

    题解:那么,在n、m<=100000时我们该怎么做呢,很明显,我们可以用线段树,但是我还没调出来。。。也许明天会记得填坑吧,大概。

  • 相关阅读:
    [BZOJ4698][SDOI2008]Sandy的卡片(后缀自动机)
    [NOI2015]小园丁与老司机(DP+上下界最小流)
    [BZOJ2007][NOI2010]海拔(对偶图最短路)
    [NOI2018]屠龙勇士(exCRT)
    [NOI2018]归程(可持久化并查集,Kruskal重构树)
    [BZOJ2823][BZOJ1336][BZOJ1337]最小圆覆盖(随机增量法)
    [BZOJ1069][SCOI2007]最大土地面积(水平扫描法求凸包+旋转卡壳)
    [BZOJ1143][CTSC2008]祭祀river(Dilworth定理+二分图匹配)
    [BZOJ3160]万径人踪灭(FFT+Manacher)
    [NOI2015]寿司晚宴
  • 原文地址:https://www.cnblogs.com/Beginner-/p/7226326.html
Copyright © 2011-2022 走看看