素数问题

素数:一个大于1的自然数,除了它自己和1以外没有其他因子。

问题1:给定自然数n,判断其是否为素数
一个自然数若为合数,则其必有一个不大于其平方根的因子。

//自然数n是否是素数,若是则返回true,反之,返回false
bool IsPrime(int n)
{
    if(n < 2)
        return false;

    for(int i = 2; i <= sqrt(n); ++i)
    {
        if(n % i == 0)
            return false;
    }
    return true;
}

问题2:求小于自然数n的所有素数的个数
方法一:
当自然数r为质数时,其整数倍数全为合数,如果r已经被判断不是质数了,那么再找到r后面的质数来把这个质数的倍数筛掉。
用bool类型数组flags[n]保存对应的状态位,初始化全为true,先从下标2开始,2为素数,其后面的倍数全为合数,合数对应的状态数组元素置为false;查找下一个状态为true的元素,为3,则3为素数,同理,其后面的倍数全为合数,合数对应的状态数组元素置为false,依次下去。。。

int countPrimes(int n) 
{       
        if(n <= 2)
            return 0;

        if(n == 3)
            return 1;

        int size = (n + 1) / 2;
        //动态分配标志空间(除2以外的偶数必为合数,直接排除,优化空间)
        bool *flags = new bool[size];

        int i;
        for(i = 1; i < size; ++i)
        {
            //初始化状态位,下标i中保存的是自然数2i + 1的状态
            flags[i] = true;
        }

        for(i = 1; i < size; ++i)
        {
            if(flags[i])
            {
                int j = 3 * i + 1;
                for(; j < size; j += 2 * i + 1)
                {
                    //自然数2i + 1的正整数倍数(除本身以外的)的状态位置false
                    flags[j] = false;
                }
            }
        }

        //初始化计数(状态数组未包含素数2的状态,故从1开始计数)
        int cnt = 1;
        for(i = 1; i < size; ++i)
        {
            if((flags[i]) && ((2 * i + 1) < n))
                ++cnt;
        }

        delete []flags;
        return cnt;
    }

方法二: 若自然数r是否为素数,判断不大于其平方根的素数是否为其因子即可。

int NumberOfPrime(int n)
{
    if(n <= 2)
        return 0;

    int *primes = new int[n];
    int length;

    primes[0] = 2;
    length = 1;

    int i;
    for(i = 3; i < n; ++i)
    {
        int squRoot = (int)sqrt(n * 1.0);
        bool flag = true;
        int j;
        for(j = 0; j < length; ++j)
        {
            if(primes[j] <= squRoot)
            {
                if(i % primes[j] == 0)
                {
                    flag = false;
                    break;
                }
            }
            else
                break;
        }

        if(flag)
        {
            primes[length] = i;
            ++length;
        }
    }
    delete []primes;

    return length;
}

问题3:打印前n个素数
采用问题2中的方法二

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值