素数的概念
素数即质数,是指在大于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的素数为止,现在判断一个数是不是素数,可以使用上面介绍的试除法判断素数来进行判断。那么现在如何判断一个数是不是回文数呢,
而所有的数的位数只有两种可能
-
奇数位
- 如果说原来的数是 abc 那么说反转之后就是 abcba
,我们可以很容易看到,只需要把原来的数字反转之后,去除掉第一位拼接在原来的字符串后面就是回文串,然后判断这个新得到的数是不是质数,就可以得到答案
- 如果说原来的数是 abc 那么说反转之后就是 abcba
-
偶数位
- 如果说原来的数是 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];
}
}