zoukankan      html  css  js  c++  java
  • 1015

    #define _CRT_SECURE_NO_WARNINGS
    #include <cstdio>
    #include <cmath>
    #include<iostream>
    #include<algorithm>
    using namespace std;
    /*埃拉托斯特尼筛法是一个快速获取小于数X的所有素数集合的算法。
    首先我们要明确,假设一个合数x能表示为两个数的乘积,他必定有一个小于等于sqrt(x)的因子,这可以用归谬证明法证明。如果两个因子都大于sqrt(x),那么乘积大于x,这和假设矛盾。
    所以,判断一个数x是否是合数,只要依次除以2至sqrt(x)间的素数,判断是否整除即可。
    埃拉托斯特尼筛法基于以下原理,给定一个素数n>1,kn是一个合数(k>1),例如n=3,那么6,9,12,15…都是合数。
    以100为例,我们先创建一个100个数字的数组。
    先使用最小的素数2,将所有2的倍数(除2本身)标记为合数。
    接下来2+1的数是3,此时检查3是不是素数,检查标记,发现没有被标记为合数(因为不是2的倍数),所以再将所有3的倍数标记为合数。
    下一个数是4,发现他已经被标记为合数,所以他可以表示小于4的素数的乘积2*2,所以4的倍数必定含有因子2,所以所有4的倍数已经全部被标记过,直接跳过4。
    下一个数是5,没有被标记为合数,把所有小于100的5的倍数标记为合数
    ………这样一直计算到sqrt(100),即10。
    那么为什么不标记大于10的数例如11呢?因为所有的倍数已经被标记过了,例如22,33,44,55…分别有因子2,3,2,5,大于10的倍数,例如11*11已经超过max了,参见最上面的推论
    注意这里有一个优化点,很多书籍上或者教程上都没有说出来,只要标记大于本身的倍数就行了,例如5,只要标记5*5,5*6,5*7…为合数,因为5*2,5*3,5*4…已经被之前出现的数的倍数标记过了

    #include <cstdio>
    #include <cmath>
    using namespace std;
    bool isprime(int n) {
    if(n <= 1) return false;
    int sqr = int(sqrt(n * 1.0));
    for(int i = 2; i <= sqr; i++) {
    if(n % i == 0)
    return false;
    }
    return true;
    }
    int main() {
    int n, d;
    while(scanf("%d", &n) != EOF) {
    if(n < 0) break;
    scanf("%d", &d);
    if(isprime(n) == false) {
    printf("No ");
    continue;
    }
    int len = 0, arr[100];
    do{
    arr[len++] = n % d;
    n = n / d;
    }while(n != 0);
    for(int i = 0; i < len; i++)
    n = n * d + arr[i];
    printf("%s", isprime(n) ? "Yes " : "No ");
    }
    return 0;
    }

  • 相关阅读:
    .Net下的MSMQ(微软消息队列)的同步异步调用
    [收藏]JS获取网页中HTML元素的几种方法分析
    在FireFox下设为首页的解决方法
    如何创建和使用Web Service代理类
    [收藏]61条面向对象设计的经验原则
    [总结]DotNet中用到的加密算法总结
    如何把用SQL语句int型整数转换成二进制数
    彻底杜绝PHP的session,cookie,Cannot modify header错误
    MSN总是报80048820的错误,网上搜的一些资料解决不了,我找到了真正解决办法!
    [收藏]MD5加密的javascript实现
  • 原文地址:https://www.cnblogs.com/masayoshi/p/10857925.html
Copyright © 2011-2022 走看看