*【每日一题 提高题】[蓝桥杯 2022 国 A] 选素数

选素数

小蓝有一个数 x,每次操作小蓝会选择一个小于 x 的素数 p,然后在 x 成为 p 的倍数前不断将 x 加 1,(如果 x 一开始就是 p 的倍数则 x 不变)。

小乔看到了小蓝进行了 2 次上述操作后得到的结果 n,他想知道 x 在一开始是多少。如果有多种可能,他想知道 x 一开始最小可以是多少,而如果不存在任何解,说明小乔看错了,此时请输出 −1。
输入格式

输入一行包含一个整数 n,表示经过两次操作后 x 的值。
输出格式

输出一行包含一个整数表示 x 的初始值。如果有多个解,输出最小的。如果不存在解,请输出 −1

这个题主要考察质因数分解和质数判断

首先质数的判断,这个太常见了,这里就不多赘述

java 复制代码
private static boolean isPrime(int n){
        if (n<2)return false;
        if (n==2||n==3)return true;
        if (n%6!=1&&n%6!=5)return false;
        for (int i = 5; i*i<= n ; i+=6) {
            if (n%i==0||n%(i+2)==0){
                return false;
            }
        }
        return true;
    }

然后是质因数分解,这里用了 List 集合去装分解出来的质数

java 复制代码
private static List<Integer> prime(int n){
        List<Integer> ans  = new LinkedList<>();
        for(int i = 2; i <= n/i ; i++){//判断条件用n/i,以防i*i<=n发生溢出
            int a = 0;//每次循环都要清零
            while(n % i == 0){
                n /= i;
                a++;
            }
            if(a > 0)
                ans.add(i);
        }
        //若该数是质数,那么应该将自身(n)也加入
        if(n > 1) ans.add(n);
        return ans;
    }

最后就是根据题意进行模拟,a0→a1→ a2

(依次进行第一次操作、第二次操作)。

java 复制代码
import java.util.*;//导入万能类 当然也不是很万能
public class Main {
    static List<Integer> oList = new LinkedList<>();//需要用到的集合
    public static void main(String[] args) {
    //标准输入 一个n
       Scanner sc  = new Scanner(System.in);
       int n = sc.nextInt();
       
        if (isPrime(n)){
        //如果输入数为质数,那么打印-1。因为我们的n应该是质数的倍数,且必须至少为2倍(题目中说了选择一个小于x的素数p,所以至少是2倍)。
            System.out.println("-1");
        }else {
            oList = prime(n);
            //对其质因数分解,并用集合装好
            int minstart = Integer.MAX_VALUE;//定义所求的最小开始的a0,赋值为Integer最大值,为了后面获取最小值
            for (int i = 0; i < oList.size(); i++) {
            //遍历质因数集合
                int last_left = n-oList.get(i);
                //下面对第一次操作后可能的数组遍历
                for (int j = last_left+1; j <= n ; j++) {
                //这里的范围为什么是从last_left+1到n
                //假设第二次操作选择的是oList.get(i)作为质数,那么第二次操作前的数(a1)不会小于n-oList.get(i)
                //上次操作前的数字(a1)必然在左边界加1和n之间。(为什么是n?因为n可能本身就是某个质数的倍数,然后这次操作又选了该质数,则n不变)
                    if (isPrime(j)){
                        continue;
                        //跟上面一样,不可能是质数。
                    }
                    List<Integer> temp = prime(j);//继续对上个数进行质因数分解
                    int max = Collections.max(temp);
                    //找出最大的质因数作为第一次操作选择的质数,这样能使起始数字(即a0)尽可能小
                    minstart = Math.min(j-max+1,minstart);//更新维护最小数字
                   
                }
            }
            if (minstart==Integer.MAX_VALUE){
                System.out.println("-1");
            }else {
                System.out.println(minstart);
            }
        }

    }
    //质数的判断
    private static boolean isPrime(int n){
        if (n<2)return false;
        if (n==2||n==3)return true;
        if (n%6!=1&&n%6!=5)return false;
        for (int i = 5; i*i<= n ; i+=6) {
            if (n%i==0||n%(i+2)==0){
                return false;
            }
        }
        return true;
    }
    //对n进行质因数分解
    private static List<Integer> prime(int n){
        List<Integer> ans  = new LinkedList<>();
        for(int i = 2; i <= n/i ; i++){//判断条件用n/i,以防i*i<=n发生溢出
            int a = 0;//每次循环都要清零
            while(n % i == 0){
                n /= i;
                a++;
            }
            if(a > 0)
                ans.add(i);
        }
        //若该数是质数,那么应该将自身(n)也加入
        if(n > 1) ans.add(n);
        return ans;
    }
}
相关推荐
铲子Zzz8 分钟前
Java使用接口AES进行加密+微信小程序接收解密
java·开发语言·微信小程序
霖檬ing13 分钟前
K8s——配置管理(1)
java·贪心算法·kubernetes
小小小新人1212328 分钟前
C语言 ATM (4)
c语言·开发语言·算法
你的冰西瓜1 小时前
C++排序算法全解析(加强版)
c++·算法·排序算法
Vic101011 小时前
Java 开发笔记:多线程查询逻辑的抽象与优化
java·服务器·笔记
Biaobiaone1 小时前
Java中的生产消费模型解析
java·开发语言
এ᭄画画的北北1 小时前
力扣-31.下一个排列
算法·leetcode
特立独行的猫a1 小时前
11款常用C++在线编译与运行平台推荐与对比
java·开发语言·c++
louisgeek2 小时前
Java 位运算
java