纪中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

 

posted @ 2019-08-09 16:19  Vanilla_chan  阅读(270)  评论(0编辑  收藏  举报