zoukankan      html  css  js  c++  java
  • 纪中9日T4 2298. 异或

    2298. 异或 

    (File IO): input:gcdxor.in output:gcdxor.out

    时间限制: 1000 ms  空间限制: 262144 KB  具体限制  

    题目描述

    SarvaTathagata是个神仙,一天他在研究数论时,书上有这么一个问题:求不超过n两两的数的gcd。
    SarvaTathagata这么神仙的人当然觉得这个是sb题啦。学习之余,他还发现gcd的某一个特别好的性质:如果有两个数i,j满足gcd(i,j)=i^j(这里的^为c++中的异或)的话,那么这两个数组成的数对(i,j)就是一个nb的数对(这里认为(i,j)和(j,i)为相同的,并不需要计算2次)。
    当然,SarvaTathagata并不会只满足于判断一个数对是否nb,他还想知道满足两个数都是不超过n并且nb的数对有多少个。
    由于SarvaTathagata实在是太神仙了,他认为这种题实在是太简单了。于是他找到了你,看看你是否能解决这个问题。

    输入

    共一行一个整数n,含义如题所述。

    输出

    一行一个整数,表示nb的数对的个数。

    样例输入

    样例输入1

    12

    样例输入2

    123456

    样例输出

    样例输出1

    8

    样例输出2

    214394

    数据范围限制

     

    提示

    样例1中共有八对,分别是: {1,3},{1,5},{1,7},{1,9},{2,6},{1,11},{2,10},{4,12}。

    提示有误,特此隐藏


    Solution

    这是一道数论题,涉及gcd,xor(^),二进制减法。

    以上三者中,若你对任何一样过敏,请谨慎食用。

    Way one(40分)

    用个程序找规律

    //gcdxor table
    #include<bits/stdc++.h>
    using namespace std;
    int gcd(int ta,int tb)
    {
        if(tb==0) return ta;
        if(ta%2!=0&&tb%2!=0) return gcd(tb,ta%tb);
        if(ta%2==0&&tb%2!=0) return gcd(ta/2,tb);
        if(ta%2!=0&&tb%2==0) return gcd(ta,tb/2);
        return 2*gcd(ta/2,tb/2);
    }
    
    int ans[10000001];
    int n;
    bool vis[10000001],memory[1001][1001];//memory[i][j]
    int search(int num)
    {
        if(vis[num]) return ans[num];
        if(num==1)
            return 0; 
            
        vis[num]=1;
        int now=0;
        for(int i=1;i<num;i++)
        {
            if(gcd(num,i)==(num xor i))
                now++;
    //        if(gcd(i,num)==(i xor num))
    //            now++;
            
        }
    //    if(gcd(num,num)==(num xor num)) 
    //        now++;
        ans[num]=ans[num-1]+now;
        return ans[num];
    }
    int main()
    {
        freopen("table4.txt","w",stdout);
        cin>>n;
    //    int s=clock();
        int k;
        /*
        for(k=0;k<=n;k++)
        {
            if(k==0) {cout<<0<<endl;continue;}
            for(int i=1;i<=k;i++)
            {
                for(int j=1;j<=k;j++)
                {
                    if(gcd(i,j)==(i xor j)){
                        memory[i][j]=true;
                        ans[k]++;
                        
                    }
    //                cout<<memory[i][j]<<" ";
                }
    //            cout<<endl;
            }
            
            cout<<k<<" "<<ans[k]/2<<endl;
        }
        */
    //    /*
        for(int k=0;k<=n;k++)
        {
            if(k==0) {cout<<0<<endl;continue;}
            cout<<k<<" "<<search(k)<<","<<endl;
        }
    //    */
    //    int e=clock();
    //    cout<<e-s;
    
    
        return 0;
    }
    可以无视我

    找不到……

    Code(40分)

    #pragma GCC optimize(2)
    #include<bits/stdc++.h>
    #define IL inline
    using namespace std;
    short int diff[2001]{0,0,0,1,0,1,1,1,0,1,1,1,1,1,1,3,0,1,1,1,1,1,1,1,1,1,1,3,1,1,3,1,0,1,1,1,1,1,1,2,1,1,1,1,1,3,1,1,1,1,1,3,1,1,3,2,1,1,1,1,3,1,1,5,0,1,1,1,1,1,1,1,1,1,1,2,1,1,2,1,1,1,1,1,1,3,1,2,1,1,3,1,1,1,1,3,1,1,1,3,1,1,3,1,1,1,1,1,3,1,2,3,1,1,1,1,1,1,1,3,3,1,1,3,1,3,5,1,0,1,1,1,1,1,1,3,1,1,1,1,1,1,1,3,1,1,1,2,1,1,2,1,1,3,1,1,2,1,1,2,1,1,1,1,1,3,1,1,1,1,3,3,1,1,2,4,1,1,1,1,3,1,1,2,1,1,1,3,1,3,3,1,1,1,1,3,1,1,3,1,1,1,1,1,3,2,1,4,1,1,1,1,1,1,1,2,3,1,1,3,2,3,3,1,1,1,1,1,1,1,1,4,1,1,1,1,1,1,3,1,3,1,1,3,1,3,3,2,1,1,3,1,5,1,1,7,0,1,1,1,1,1,1,1,1,1,1,2,1,1,3,1,1,1,1,1,1,1,1,2,1,1,1,1,1,2,3,2,1,1,1,2,1,1,2,2,1,3,1,1,2,1,1,2,1,1,3,1,1,1,1,1,2,1,1,4,1,1,2,3,1,1,1,1,1,3,1,2,1,1,3,1,1,2,1,3,1,1,1,2,3,1,3,2,1,1,1,1,2,1,4,5,1,1,1,1,1,1,1,1,3,1,1,4,1,3,2,1,1,1,1,1,1,1,3,3,1,1,3,1,3,1,1,1,1,1,1,3,1,1,3,1,1,1,1,1,3,1,1,4,1,1,1,1,1,2,1,1,3,1,2,3,1,1,4,2,1,1,1,1,1,1,1,2,1,1,1,1,1,3,2,1,3,1,1,3,1,1,3,1,2,3,3,1,3,2,1,3,1,1,1,1,1,1,1,4,1,1,1,3,1,1,4,1,1,1,1,1,1,1,1,2,1,1,1,3,3,2,1,1,3,1,1,3,1,3,3,1,1,1,3,1,3,1,2,10,1,1,1,1,3,1,1,1,5,1,1,3,1,1,7,3,0,1,1,1,1,1,1,2,1,1,1,1,1,2,1,1,1,1,1,2,1,1,2,2,1,1,1,2,3,1,1,2,1,1,1,1,1,1,1,1,1,1,1,2,1,1,2,3,1,3,1,1,1,2,1,4,1,1,2,1,3,1,2,4,1,1,1,2,1,1,2,1,1,3,1,1,2,1,2,2,1,1,3,2,1,1,1,1,2,1,1,4,1,2,2,1,1,1,1,1,3,1,1,3,1,1,1,1,1,3,1,2,2,1,1,4,1,3,4,1,1,1,1,1,2,3,3,4,1,1,1,1,1,3,1,1,1,1,3,2,1,1,2,3,1,1,1,1,3,1,1,3,1,1,2,1,1,1,3,2,1,1,1,2,1,1,2,2,3,1,1,1,3,3,2,2,1,1,1,1,1,3,1,2,2,3,1,2,4,1,5,3,1,1,1,1,1,1,1,2,1,1,1,3,1,1,1,1,3,1,1,2,1,3,4,1,1,3,3,2,2,1,1,7,1,1,1,1,1,1,1,1,1,1,1,3,3,1,3,1,1,1,1,1,3,1,1,5,3,1,1,1,1,7,1,3,1,1,1,3,1,1,3,2,1,1,1,1,3,1,1,4,1,1,1,1,1,1,1,2,3,1,1,3,1,1,4,2,1,1,1,1,1,2,1,2,1,1,2,1,1,1,1,2,3,1,1,3,2,1,3,1,1,3,1,1,4,1,2,3,1,1,1,1,1,1,1,1,1,1,1,2,1,4,2,4,1,1,1,1,1,1,1,4,1,1,3,1,2,2,1,1,3,1,1,3,1,1,3,2,1,3,1,2,3,1,1,3,2,1,3,1,3,2,1,1,3,1,2,9,1,1,3,2,1,1,1,1,1,1,1,4,1,1,1,1,1,2,4,1,1,1,1,2,1,1,3,1,1,1,1,1,4,3,1,3,1,1,1,1,1,1,1,2,1,1,1,2,1,1,2,2,1,1,1,1,1,1,3,2,3,1,2,1,1,3,1,3,3,1,1,3,1,3,3,1,1,1,3,1,3,1,1,9,1,1,1,1,3,1,1,1,3,1,1,3,2,1,10,1,1,1,1,1,1,1,1,3,3,1,1,1,1,3,1,1,5,1,1,3,1,1,3,5,1,3,1,1,7,1,3,7,0,1,1,1,1,1,1,1,1,1,1,3,1,1,2,1,1,1,1,1,1,2,1,2,1,1,2,1,1,3,1,2,1,1,1,2,1,1,2,1,1,1,1,2,2,1,2,4,1,1,1,1,1,1,2,1,3,1,1,4,1,2,2,1,1,1,1,1,1,1,1,3,1,1,1,1,1,1,1,1,1,3,1,2,1,1,2,1,1,1,1,1,2,1,3,2,1,1,3,1,1,2,1,2,1,1,2,2,1,1,4,2,1,1,1,3,2,1,1,2,3,1,1,1,2,1,4,1,1,1,1,2,1,1,2,1,1,3,1,1,2,2,1,2,1,1,3,1,1,2,1,2,2,1,1,4,2,1,2,3,1,1,1,1,3,1,2,2,1,1,1,1,1,3,1,2,2,1,1,2,1,2,4,2,1,1,2,1,2,1,1,8,1,1,1,1,1,1,1,1,3,1,1,2,1,1,3,1,1,1,1,3,1,1,1,4,1,3,3,2,1,2,2,2,2,1,1,2,1,1,4,2,1,1,3,1,4,3,1,2,1,1,1,1,1,1,1,1,2,1,3,6,3,1,4,1,1,1,1,1,1,3,1,2,1,1,3,1,1,1,1,4,1,1,1,2,3,1,2,1,1,2,1,1,2,2,3,4,1,1,1,1,1,1,1,1,3,1,1,3,1,3,3,1,1,1,1,1,2,1,1,3,1,1,1,1,3,2,2,2,1,1,1,2,1,1,2,2,1,1,1,1,2,1,2,3,3,1,1,1,1,5,1,1,3,1,3,2,2,1,2,5,1,1,1,1,1,1,1,2,1,1,3,2,1,1,2,3,2,1,3,2,1,1,2,1,4,1,1,1,5,3,3,5,1,1,1,1,1,1,1,2,1,1,1,4,1,1,2,1,1,1,1,1,1,1,3,2,1,1,1,1,1,1,1,1,3,1,1,2,1,3,2,1,1,3,3,1,4,1,1,5,1,1,3,1,3,1,2,4,2,1,1,4,1,2,7,1,1,1,1,1,1,1,1,2,1,1,1,1,1,3,1,1,1,1,1,2,1,1,3,4,3,1,1,1,3,1,1,3,1,1,1,1,1,1,1,1,3,1,1,2,1,3,5,2,3,1,1,1,1,3,1,2,1,1,7,1,1,3,3,3,1,1,1,3,1,1,3,1,1,1,1,1,3,1,2,4,1,1,1,1,1,1,1,1,3,1,1,3,1,2,4,1,1,1,1,1,1,1,1,5,1,1,1,1,1,1,2,1,3,1,1,3,1,1,3,2,1,2,1,2,4,1,2,6,1,1,1,1,1,2,1,1,1,1,2,3,1,1,2,2,1,1,1,1,2,1,1,2,1,3,1,1,1,2,2,2,3,1,1,3,1,1,3,1,2,1,1,1,3,2,1,4,1,3,3,1,1,1,1,2,4,1,1,3,2,1,3,1,1,1,1,1,1,1,1,2,1,1,1,1,1,3,1,1,1,1,1,3,1,2,2,2,1,1,4,2,2,1,4,4,1,1,1,1,1,1,1,2,1,1,1,2,1,1,4,1,1,1,1,3,3,2,1,2,2,1,2,1,1,3,1,3,3,1,1,3,1,1,3,2,1,3,1,1,3,1,2,4,1,1,3,1,1,1,2,2,3,1,1,7,1,1,3,1,2,1,1,1,3,2,1,2,3,1,2,3,1,3,1,3,3,1,1,3,2,1,9,1,1,3,1,1,3,1,2,5,1,1,1,1,1,1,1,3,1,1,1,2,1,2,4,2,1,1,1,1,1,1,1,3,1,1,2,2,4,1,1,1,1,1,1,2,1,1,2,1,1,1,1,1,3,1,1,2,1,1,1,2,1,2,1,1,4,1,3,2,1,2,3,6,1,1,1,1,1,1,1,3,1,1,1,1,1,1,2,1,1,1,1,2,1,1,2,1,1,2,1,1,2,5,2,3,1,1,1,1,1,1,1,2,1,1,1,3,3,1,2,2,3,1,1,1,2,1,1,7,1,1,3,1,1,2,3,3,3,1,1,3,1,3,3,1,1,1,3,1,3,1,1,8,1,1,1,1,3,1,1,1,3,1,1,4,1,1,9,1,1,1,1,1,1,1,1,2,3,1,1,1,1,3,1,2,3,1,1,3,1,1,3,2,2,1,1,1,10,1,1,3,1,1,1,1,1,1,1,1,1,1,1,2,1,1,3,1,3};
    int n,ans;
    IL int gcd(int ta,int tb)
    {
        if(tb==0) return ta;
        if(ta%2!=0&&tb%2!=0) return gcd(tb,ta%tb);
        if(ta%2==0&&tb%2!=0) return gcd(ta/2,tb);
        if(ta%2!=0&&tb%2==0) return gcd(ta,tb/2);
        return 2*gcd(ta/2,tb/2);
    }
    IL int search(int num)
    {
        if(num==2000) {
            int t=0;
            for(int i=0;i<=2000;i++) t+=diff[i];
            return t;
        }
        int prev;
        if(num>2000)
            prev=search(num-1);
        int now=0;
        for(int i=1;i<num;i++) if(gcd(num,i)==(num xor i)) now++;
        if(gcd(num,num)==(num xor num)) now++;
        return prev+now;
    }
    int main()
    {
        freopen("gcdxor.in","r",stdin);
        freopen("gcdxor.out","w",stdout);
        cin>>n;
        for(int i=0;i<=min(n,2000);i++) 
            ans+=diff[i];
        if(n<=2000)
            cout<<ans<<endl;
        else
            cout<<search(n)<<endl;
        return 0;
    }
    40分

    其实,要是我愿意我可以把表全打出来,但是这个OJ有代码长度限制(5kb)。有毒……

    Way two

    从大神视角理解这道题:

    看到gcd就去想数论嘛

    求证:若gcd(a,b)=a xor b,则gcd(a,b)=a-b (a≥b)

    转到完整证明

    证明:

    更相减损法,可知gcd(a,b)=gcd(b,a-b)

    这个的原理与辗转相除法类似,只是把求模换成了减法

    若a-b=0,则gcd(a,b)=gcd(b,a-b)=a=b

    否则a-b>0,此时gcd(a,b)=gcd(b,a-b)>a-b

    所以,gcd(a,b)≥a-b

    再分析xor运算和二进制减法

    xor:对与每一位,若是相同则为0,若是不同则得1

    1^1=0  1^0=1  0^1=1  0^0=0

    减法:对于每一位,若是相同则得0,若是1 0则得1,若是0 1则得1并使上一位-1

    1-1=0  1-0=1  0-1=-1  0-0=0

    那么,如果两数a,b中出现了0->1的情况,则减法会使那一位上出现退位,异或(xor)则不会

    所以,a xor b≥a-b

    又因为(前面证明的)gcd(a,b)≥a-b

    所以gcd(a,b)≥a-b≥a xor b

    所以当gcd(a,b)=a xor b时,gcd(a,b)=a-b=a xor b

    所以若gcd(a,b)=a xor b,则gcd(a,b)=a-b (a≥b)

    证毕

    回到此题

    现在我已经证出来

    若gcd(a,b)=a xor b,则gcd(a,b)=a-b (a≥b)

    一看题目,就是当gcd(a,b)=a xor b时

    那么我们就可以直接用结论了

    设c=a-b,则b=a-c

    若数对(a,b)符合条件

    则c=a^b

    故我只要枚举a和b即可!

    Code(90分)

    #pragma GCC optimize(2)//这个程序不开O2会超时(90分)
    #include<bits/stdc++.h>
    using namespace std;
    int n,cnt[20000001],sum[20000001];
    int main()
    {
    //    freopen("gcdxor.in","r",stdin);
    //    freopen("gcdxor.out","w",stdout);
        cin>>n;
        memset(cnt, 0, sizeof(cnt));
        for(int c = 1; c <= n; c++)
        for(int a = c*2; a <= n; a += c) //因为a>=b,所以需要从2*c开始枚举
        {
            int b = a - c;
            if(c == (a ^ b)) cnt[a]++;//统计每个a对应的b的数量
        }//^的优先级低于==,所以要打上括号 
        sum[0] = 0;
        for(int i = 1; i <= n; i++) 
            sum[i] = sum[i-1] + cnt[i];
        cout<<sum[n];
        return 0;
    }

    Code(100分)

    #include<bits/stdc++.h>
    using namespace std;
    int n,sum;
    int main()
    {
    //    freopen("gcdxor.in","r",stdin);
    //    freopen("gcdxor.out","w",stdout);
        scanf("%d",&n);
    //    int s=clock();
        for(int c=1;c<=n;c++)//c=a-b
        for(int a=c*2;a<=n;a+=c) //因为a>=b,所以需要从2*c开始枚举
            if(c==(a ^ (a-c))) sum++;
        //统计每个a对应的b的数量
        //^的优先级低于==,所以要打上括号 
        printf("%d",sum);
    //    int e=clock();
    //    cout<<endl<<e-s;
        return 0;
    }

    TLE?

     细心的你应该发现这个:

    不要怕TLE啦

    如果你会算复杂度,会一点微积分,会一点金融学,你就要知道:

    所以整个程序的时间复杂度为O(nlog(n))

    Attention

    一、^的优先级低于==,所以要打上括号

    二、因为a>=b,所以需要从2*c开始枚举,使得a的最小值为2*c,b=a-c,b的初值值为c且越来越小

    三、用前缀和的思想,递归sum[i] = sum[i-1] + cnt[i],即n的答案是在n-1的基础上加上这次枚举的答案

    END

  • 相关阅读:
    const用法详解(转)
    [转]Scintilla开源库使用指南
    javascript 流程控制语句 if 语句
    knockout.js 练习一
    深入Node.js的模块机制
    js本地存储解决方案(localStorage与userData)
    linear-gradient 渐变 css3新属性
    制作响应式网站时,用来测试不同宽度下网页展示效果的方法
    zepto.js, django和webpy
    attachEvent 与 addEventListener 的用法、区别
  • 原文地址:https://www.cnblogs.com/send-off-a-friend/p/11327207.html
Copyright © 2011-2022 走看看