素数,不能被除了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. 记一次log4j2引发的渗透测试

    前言 记一次log4j2打入内网并用CVE-2021-42287.CVE-2021-42278获取到DC权限的靶场渗透. 外网打点 首先对web进行端口扫描,发现38080端口和22端口 访问一下38 ...

  2. mysql语句1-创建库和表

    一.DDL数据定义语言 就是对书库内部的对象进行创建.删除.修改等操作的语言. 关键字:create  drop  alter 1.连接数据库 mysql -u用户名 -p -h指定主机(不指定默认是 ...

  3. 初识python: 斐波拉契数(生成器获取)

    使用  生成器(yield) 获取斐波拉契数. 代码如下: def fun(n): a,b,c = 0,0,1 while a < n: yield b # b, c = c, b + c 以下 ...

  4. [转]Python3字符串前缀u、b、r

    1.无前缀 & u前缀 字符串默认创建即以Unicode编码存储,可以存储中文. string = 'a'  等效于  string = u'a' Unicode中通常每个字符由2个字节表示 ...

  5. centos7 alias别名永久生效

    进入/etc/profile.d/目录 cd /etc/profile.d/ 在profile.d目录随意创建一个sh文件,例如alias_test.sh vi alias_test.sh##里面的内 ...

  6. Pandas系列(十八)- 多级索引

    多级索引 多级索引(也称层次化索引)是pandas的重要功能,可以在Series.DataFrame对象上拥有2个以及2个以上的索引.实质上,单级索引对应Index对象,多级索引对应MultiInde ...

  7. [转]webpack配置本地服务器

    亲测,webpack打包vue项目之后生成的dist文件可以部署到 express 服务器上运行. 我的vue项目结构如下: 1. 进入该vue项目目录,打开git bash,执行:npm run b ...

  8. 关于APP设计规范和一些图层命名

    首先,本人大学计算机专业出身,学过编程,工作的时候做过 产品经理,设计师,前端工程师,对工作的流程都有一些见解. 现在主攻前端工程师,做Web APP.今天收到设计师的设计稿,一看图层分类,这让我感觉 ...

  9. python极简教程04:进程和线程

    测试奇谭,BUG不见. 大家好,我是谭叔. 这一场,主讲python的进程和线程. 目的:掌握初学必须的进程和线程知识. 进程和线程的区别和联系 终于开始加深难度,来到进程和线程的知识点~ 单就这两个 ...

  10. 生产环境上,哨兵模式集群Redis版本升级应用实战

    背景: 由于生产环境上所使用的Redis版本并不一致,好久也没有更新,为了避免版本不同对Redis集群造成影响,从而升级为统一Redis版本! 1.集群架构 一主两从三哨兵: 2.升级方案 (1)升级 ...