素数,不能被除了1和本身以外整除的数被称为素数。接下来我用三种方式求得1~100以内素数。

方式一

外层每循环一次,内层就计算出这个数有几个因子,我们都知道素数的因子只有两个,所以如果个数为2就加进总和里面:

package day_11_25;

/**
* 计算1-100之间的素数和
*
* @author soberw
*/ public class PrimeFor {
public static void main(String[] args) {
//记录和
int sum = 0;
//记录因子个数
int count = 0;
int counter = 0;
for (int i = 2; i <= 100; i++) {
//初始置0
count = 0;
for (int j = 1; j <= i; j++) {
counter++;
if (i % j == 0) {
count++;
}
}
//两个因子为素数
if (count == 2) {
sum += i;
} }
System.out.println("总和为" + sum);
System.out.println("循环了" + counter + "次");
} }

运行结果:

共计算了5049次。

方式二

方式一虽然好理解,但是也存在很多的问题,比如如果一个数他本来就是偶数(当然除了2),那就没有判断的必要了,也还有就是没有中断条件,就算已经知道了这个数不是素数了,但程序还是从头到尾循环了一遍,于是我做了改进,加入了互斥锁,并且从2开始计算(因为最小素数为2),实现如下:

package day_11_25;

/**
* 计算1-100之间的素数和
*
* @author soberw
*/ public class PrimeFor2 {
public static void main(String[] args) {
int sum = 0;
//声明互斥锁
boolean flag = true;
int counter = 0;
for (int i = 2; i <= 100; i++) {
flag = true;
//偶数直接跳到下一次
if (i != 2 && i % 2 == 0){
continue;
}
//从2到除了它本身的数之间判断
for (int j = 2; j < i; j++) {
counter++;
//有因子直接退出
if (i % j == 0) {
flag = false;
break;
}
}
if (flag) {
sum += i;
}
}
System.out.println("总和为" + sum);
System.out.println("循环了" + counter + "次");
} }

运行结果:

相比于第一种方法确实快了不少,共计算了1084次。

方式三

那么有没有更好的方法呢,答案是肯定的,我们在判断的时候,都是从头到尾去循环一遍,就算是加了互斥锁,也要一次加一个去判断,有点繁琐。
那有没有更好的解决方式呢,于是我就想到了下面这种方法,通过开平方判断。打个比方,如果我们要判断100是不是素数,就首先确定一个中间数,你可以找到100的根(10),将数分成两份,如图:

图可能画的有点抽象,其实就是我们将10作为中间数,10前面的数乘以10后面的数如果有出现等于100的情况,那就不是素数(比如2x50=100)。
因为因子都是成对存在的,1和100,2和50,4和25,5和20,10和10。成对的因子,其中一个必然小于等于100的开平方,另一个大于等于100的开平方。所以这样一来我们就最多判断10次就行了,一下子减少了90次。效率成倍提高。而且实现起来也不复杂,如下:

package day_11_25;

/**
* 计算1-100之间的素数和
*
* @author soberw
*/ public class PrimeFor3 {
public static void main(String[] args) {
int sum = 0;
int counter = 0;
label:
for (int i = 2; i <= 100; i++) {
if (i != 2 && i % 2 == 0) {
continue;
}
for (int j = 2; j <= Math.sqrt(i); j++) {
counter++;
if (i % j == 0) {
continue label;
}
}
sum += i;
}
System.out.println(sum);
System.out.println(counter);
} }

运行结果:

仅仅计算了187次,相比于前两种方法,大大的提高了效率。

Java经典案例之用三种方法求1~100以内素数之和的更多相关文章

  1. QThread多线程编程经典案例分析(三种方法,解释了为什么使用moveToThread的根本原因,即为了避免调用QThread::exec() )

    传统的图形界面应用程序都只有一个线程执行,并且一次执行一个操作.如果用户调用一个比较耗时的操作,就会冻结界面响应. 一个解决方法是按照事件处理的思路: 调用 Void QApplication::pr ...

  2. Java遍历List集合的三种方法

    Java遍历List集合的三种方法 List<String> list = new ArrayList<String>(); list.add("aaa") ...

  3. Java原来如此-遍历Map的三种方法

    import java.util.HashMap; import java.util.Iterator; import java.util.Map; import java.util.Set; pub ...

  4. (转)JAVA 调用Web Service的三种方法

    1.使用HttpClient用到的jar文件:commons-httpclient-3.1.jar方法:预先定义好Soap请求数据,可以借助于XMLSpy Professional软件来做这一步生成. ...

  5. Java实现ping功能的三种方法及Linux的区分

    前大半部份转自:https://blog.csdn.net/futudeniaodan/article/details/52317650 检测设备的运行状态,有的是使用ping的方式来检测的.所以需要 ...

  6. Java基础—实现多线程的三种方法

    Java虚拟机(JVM,是运行所有Java程序的抽象计算机,是Java语言的运行环境)允许应用程序并发地运行多个线程.在Java语言中,多线程的实现一般有以下三种方法: 1.实现Runnable接口, ...

  7. JAVA之线程同步的三种方法

    最近接触到一个图片加载的项目,其中有声明到的线程池等资源需要在系统中线程共享,所以就去研究了一下线程同步的知识,总结了三种常用的线程同步的方法,特来与大家分享一下.这三种方法分别是:synchroni ...

  8. JAVA中创建线程的三种方法及比较

    JAVA中创建线程的方式有三种,各有优缺点,具体如下: 一.继承Thread类来创建线程 1.创建一个任务类,继承Thread线程类,因为Thread类已经实现了Runnable接口,然后重写run( ...

  9. Java中终止线程的三种方法

    终止线程一般建议采用的方法是让线程自行结束,进入Dead(死亡)状态,就是执行完run()方法.即如果想要停止一个线程的执行,就要提供某种方式让线程能够自动结束run()方法的执行.比如设置一个标志来 ...

随机推荐

  1. 分布式链路追踪自从用了SkyWalking,睡得真香!

    本篇文章介绍链路追踪的另外一种解决方案Skywalking,文章目录如下: 什么是Skywalking? 上一篇文章介绍了分布式链路追踪的一种方式:Spring Cloud Sleuth+ZipKin ...

  2. python_接口自动化测试_处理参数替换

    在进行自动化测试时,通常会存在A接口用例的返回值是B接口用例的入参这样的情况 可进行如下方式处理: step1.处理A用例时,在响应结果中提取出该数据的值,并赋给一变量,比如 exeId = res. ...

  3. Java复制文件用数据流方法,renameTO()方法是相当于剪切操作

    我想达到的效果是,一个文件复制到另一个地方,然后重命名 //判断是否存在 File file = new File("D:/tomcat9.0.12/apache-tomcat-9.0.12 ...

  4. Linux上天之路(三)之Linux系统目录

    1. Linux设计思想 1) 程序应该小而专一,程序应该尽量的小,且只专注于一件事上,不要开发那些看起来有用但是90%的情况都用不到的特性: 2) 程序不只要考虑性能, 程序的可移植性更重要,she ...

  5. docker 修改容器env配置

    docker 修改容器env配置 场景:修改zabbix数据库密码 zabbix容器构成: 数据库:zabbix-mysql server端:zabbix-server-mysql web端:zabb ...

  6. 【分享数据】vm-insert的压缩比达到29倍

    vm-insert采用remote-write的http协议来接收metric数据,然后按照一定算法转发到vm-storage群集. vm-insert到vm-storage这里是用了自己的二进制协议 ...

  7. VictoriaMerics学习笔记(1):翻译官方广告

    先看看VictoriaMetrics官网网站上是如何作(tree)宣(new)传(bee)的: 官方广告 0.(监控领域)最快解决方案 为高性能而设计 便于安装 支持单机和群集版本 1.更高效的存储空 ...

  8. Ultimaker2+使用指南

    USB打印 使用最新的官方软件即可进行USB3D打印. 使用时在选项上进行预热以及x,y,z重新归零定位,每次开始都要做. SD卡打印 比较方便,因为电脑可能拿去做其他事情. 打印机堵塞 退出耗材时候 ...

  9. 安卓开发之intent

    两个活动之间的跳转要通过intent来进行,intent跳转分为隐式的和显示的. 首先xml中定义Button,通过按下按钮实现回调,在回调函数中进行相应intent设置. <Button an ...

  10. /usr/local /opt

    Linux 的软件安装目录是也是有讲究的,理解这一点,在对系统管理是有益的 /usr:系统级的目录,可以理解为C:/Windows/, /usr/lib理解为C:/Windows/System32. ...