定义:对于a、b两个整数,如果a和b的差能被m整除,那么a和b关于模m同余,记作a≡b(mod m)

对于整数a,b,c和自然数m、n,模m同余满足:

1、自反性:a≡a(mod m);

2、对称性:若a≡b(mod m),则b≡a(mod m);

3、传递性:若a≡b(mod m),b≡c(mod m),则a≡c(mod m);

4、同加性:若a≡b(mod m),则a+c≡b+c(mod m);

5、同乘性:若a≡b(mod m),则a*c≡b*c(mod m)

       若a≡b(mod m),c≡d(mod m),则a*c≡b*d(mod m);

6、同幂性:若a≡b(mod m),则a≡b(mod m);

7、推论1:a*b mod k=(a mod k)*(b mod k)mod k;

8、推论2:若a mod p=x,a mod q=x,p、q互质,则a mod p*q=x

  证明:设a div p=k1,b div q=k2

    ∴a=p*k1+x,b=k2*q+x

    ∴p*k1=q*k2

    ∵p、q互质

    ∴k1÷q必定为整数,即q|k1

    ∴a=k*p*q+x

例1:

  形如4n+1的数称为H数,乘法在H数是封闭的。在这个集合中只能被1和本身整数整除的数称为H-素数,其余为H-合数。定义H-合成数为能且只能分解为2个H-素数的H-合数,求0~h中H-合成数的个数。  

  我们先考虑如何快速筛出H-素数,由于乘法在H数内封闭进行,并且有以下结论:

  1、两个H数乘起来一定是H数

  证明:我们设第一个H数等于4a+1,另一个为4b+1

    ∴(4a+1)(4b+1)=16ab+4a+4b+1=4(4ab+a+b)+1

    ∴乘积为H数

  2、对于一个H-合数,必定可以分解为若干个H-素数的乘积。

  3、对于一个H-素数,它的倍数一定是H-合数

  有了以上结论,我们就可以利用类似线性筛的方法,把H-素数筛出来,不过还有一点需要解决,因为乘法封闭进行,因此H-素数的倍数也不是正常意义上的倍数,只是(4n+1),其中n为正整数。

代码

#include<iostream>
#include<cstdio>
using namespace std;
const int N=1e6+10;

int prime[100005],sum[N];
bool v[N],is[N];
int main()
{
    int cnt=0;
    for(int i=5;i<N;i+=4)
    {
        if(!v[i])prime[++cnt]=i;
        for(int j=i*5;j<N;j+=i*4)
            v[j]=1;
    }
//    cout<<cnt<<endl;
    for(int i=1;i<=cnt;i++)
        for(int j=1;j<=i&&prime[i]*prime[j]<N;j++)
            is[prime[i]*prime[j]]=1;
    for(int i=1;i<N;i++)
        sum[i]=sum[i-1]+is[i];
    int x;
    while(~scanf("%d",&x)&&x)
        printf("%d %d\n",x,sum[x]);
}
02-13 15:35