[万人千题]第七讲 素数判断 题解

素数的概念

素数即质数,是指在大于1的自然数中,除了1和它自身外,不能被其他自然数整除的数。

试除法判断素数

首先,一个数的因数都是成对出现的:例如12的因数有3和4,2和6;

所以我们可以只枚举每一对中较小的那一个,假设较小的为d,较大的为n/d;

所以说 :d<(n/d) 那么进行化简 d^2<=n 最终得到的就是d<=sqrt(n);时间复杂度就从O(n) 优化成为了 O(sqrt(n)

时间复杂度是:O(sqrt(n))

bool judge(int n ){

    if(n<2)  return false;//质数是从2开始的
    
    for(int i=2;i<=n/i;i++){
    
        if(n%i==0) return false;
    }
    return true;
}

例题解析

题目来自于leetcode ,题目讲解来自Acwing,同时欢迎大家加入Acwing社区

回文素数

题目描述

求出大于或等于 N 的最小回文素数。

回顾一下,如果一个数大于 1,且其因数只有 1 和它自身,那么这个数是素数。

例如,2,3,5,7,11 以及 13 是素数。

回顾一下,如果一个数从左往右读与从右往左读是一样的,那么这个数是回文数。

例如,12321 是回文数

思路

我们可以从最小的回文数开始逐一枚举,直到找到一个大于等于N的素数为止,现在判断一个数是不是素数,可以使用上面介绍的试除法判断素数来进行判断。那么现在如何判断一个数是不是回文数呢,

而所有的数的位数只有两种可能

  1. 奇数位

    • 如果说原来的数是 abc 那么说反转之后就是 abcba
      ,我们可以很容易看到,只需要把原来的数字反转之后,去除掉第一位拼接在原来的字符串后面就是回文串,然后判断这个新得到的数是不是质数,就可以得到答案
  2. 偶数位

    • 如果说原来的数是 ab ,那么说反转之后就是 abba, 我们可以看到奇数位和偶数位相加之后,是相等的,而对于这种数而言,一定是11的倍数,也就是说一定不是质数,也就没有判断的必要了。

代码

class Solution {
    int get(int x){
        String a  = String.valueOf(x);
        String b = new StringBuilder(a).reverse().toString();
        return Integer.valueOf(a+b.substring(1));
    }
    public boolean isPrime(int x){
        if(x<2) return false;
        for(int i = 2;i<=x/i;i++){
            if(x%i==0){
                return false;
            }
        } 
        return true;
    }
    public int primePalindrome(int n) {
        if(n>7 && n<=11){
            return 11;
        }
        for(int i = 1;true;i++){
            int  x  = get(i);
            if(x>=n && isPrime(x)){
                return x;
            }
        }       
    }
}

丑数

题目描述

判断一个数是不是丑数

思路

只需要判断把 2 3 5 每个因子除干净之后,判断n是不是1即可

代码

class Solution {
    public boolean isUgly(int n) {
        if(n <= 0 ) return false;
        while(n % 2  ==0 ) n/=2;  
        while(n % 3 == 0)  n/=3;
        while(n % 5 == 0)  n/=5;
        return n==1;
    }
}

丑数 II

题目描述

我们把只包含质因子 2、3 和 5 的数称作丑数(Ugly Number)。求按从小到大的顺序的第 n 个丑数。

思路

s1 = 1*2,2*2,3*2,4*2 

s2 = 1*3,2*3,3*3,4*3

s3 = 1*5,2*5,3*5,4*5

ans = 1  U s1 U s2 U s3

因为题目中说明,丑数只包含质因子2、3、5,所以我们很容易得到上面的4个集合,注意题目中说明了1是第一个丑数,丑数实际上是就是这几个集合的并集,现在关键是要从小到大进行排列,我们可以用三个指针来移动这三个集合,每个集合的后一项实际上就是前一项的基础之上乘以相应的倍数,我们每次只需要把三个指针所指向的位置中最小的一个放入数组中就行了。

代码

class Solution {
    public int nthUglyNumber(int n) {
        int[] res = new int[n];
        int i ,j, k;
        i = j = k =0; 
        res[0] = 1;
        for(int u = 1;u<n;u++){
            int t  = Math.min(res[i]*2,Math.min(res[j]*3,res[k]*5));
            res[u] =t;
            if(t == res[i] * 2) i++;
            if(t == res[j] * 3) j++;
            if(t == res[k] * 5) k++;
        }
        return res[n-1];
    }
}
  • 4
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

佩奇inging

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值