首页 > 编程语言 >Java开发手册中-避免Random实例被多线程使用、多线程下Random与ThreadLoacalRandom性能对比

Java开发手册中-避免Random实例被多线程使用、多线程下Random与ThreadLoacalRandom性能对比

时间:2024-07-17 16:20:58浏览次数:14  
标签:ThreadLoacalRandom ThreadLocalRandom Random 实例 测试 import 多线程

场景

Java中使用JMH(Java Microbenchmark Harness 微基准测试框架)进行性能测试和优化:

https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/131723751

参考以上性能测试工具的使用。

Java开发手册中有这样一条:

【推荐】避免 Random 实例被多线程使用,虽然共享该实例是线程安全的,但会因竞争同一 seed导致的性能下降。

说明:Random 实例包括 java.util.Random 的实例或者 Math.random()的方式。

正例:在 JDK7 之后,可以直接使用 API ThreadLocalRandom,而在 JDK7 之前,

需要编码保证每个线程持有一个单独的 Random 实例。

 

Math.random() 底层确实是使用 Random 类来实现的。‌

Math.random()在底层调用了java.util.Random类的nextDouble()方法。

Random类的核心在于其内部维护了一个种子(seed),这个种子是随机数生成算法的起点。

相同的种子会生成相同的随机数序列。在创建Random对象时,如果不提供种子,则默认使用

系统时间作为种子,因此每次创建新的Random对象时,由于系统时间的不同,生成的随机数序列

也会不同。

而Random 在多线程下会因为竞争种⼦(seed)而导致性能下降,而ThreadLocalRandom 则不会出现这种情况,

因此在多线程环境下 ThreadLocalRandom 的性能会比Random 好很多。

注:

博客:
https://blog.csdn.net/badao_liumang_qizhi

实现

接下来我们来测试⼀下 ThreadLocalRandom 和 Random 性能差异,测试代码如下:

import org.openjdk.jmh.annotations.*;
import org.openjdk.jmh.runner.Runner;
import org.openjdk.jmh.runner.RunnerException;
import org.openjdk.jmh.runner.options.Options;
import org.openjdk.jmh.runner.options.OptionsBuilder;
import java.util.Random;
import java.util.concurrent.ThreadLocalRandom;
import java.util.concurrent.TimeUnit;

//测试完成时间
@BenchmarkMode(Mode.AverageTime)
//设置统计结果的时间单位
@OutputTimeUnit(TimeUnit.NANOSECONDS)
@Warmup(iterations = 2,time = 1,timeUnit = TimeUnit.SECONDS)
//测试次数和时间,参数同上
@Measurement(iterations = 5,time = 1,timeUnit = TimeUnit.SECONDS)
//fork一个线程,进行 fork 的次数,可用于类或者方法上。如果 fork 数是 2 的话,则 JMH 会 fork 出两个进程来进行测试。
@Fork(1)
//通过 State 可以指定一个对象的作用范围,JMH 根据 scope 来进行实例化和共享操作。@State 可以被继承使用,
//Scope.Thread:默认的 State,每个测试线程分配一个实例
@State(Scope.Thread)
public class ThreadLocalRandomTest {
    public static void main(String[] args) throws RunnerException {
        //启动基准测试
        Options options = new OptionsBuilder()
                .include(ThreadLocalRandomTest.class.getSimpleName())//要导入的测试类
                .build();
        new Runner(options).run();//执行测试
    }

    @Benchmark
    public void randomTest(){
        int count = 0;
        Random random = new Random();
        for (int i = 0; i < 1000; i++) {
            count += random.nextInt(10);
        }
    }

    @Benchmark
    public void threadLocalRandomTest(){
        int count = 0;
        ThreadLocalRandom localRandom = ThreadLocalRandom.current();
        for (int i = 0; i < 1000; i++) {
            count += localRandom.nextInt(10);
        }
    }
}

测试结果:

//Benchmark                                    Mode  Cnt      Score      Error  Units
//ThreadLocalRandomTest.randomTest             avgt    5  10221.461 ± 1329.015  ns/op
//ThreadLocalRandomTest.threadLocalRandomTest  avgt    5   2887.904 ±  472.645  ns/op

从上述结果可以看出ThreadLocalRandom的性能⽐Random的性能⼤约⾼ 3.5倍,

因此在多线程情况下要尽量使⽤ ThreadLocalRandom。

标签:ThreadLoacalRandom,ThreadLocalRandom,Random,实例,测试,import,多线程
From: https://www.cnblogs.com/badaoliumangqizhi/p/18307676

相关文章

  • Java - 多线程
    ThreadLocalhttps://www.cnblogs.com/fsmly/p/11020641.html为了线程安全,每个线程改的都是自己本地的副本,从而不会影响其它线程。多线程访问同一个共享变量的时候容易出现并发问题,特别是多个线程对一个变量进行写入的时候,为了保证线程安全,一般使用者在访问共享变量的时候需要......
  • Java多线程:wait/notify机制-实现生产者消费者(4)
    ......
  • java八股复习指南-多线程篇
    多线程线程的实现在Java中,实现多线程的主要有以下四种继承Thread类,重写run()方法;实现Runnable接口,实现run()方法,并将Runnable实现类的实例作为Thread构造函数的参数target;实现Callable接口,实现call()方法,然后通过FutureTask包装器来创建Thread线程;......
  • Linux多线程
    目录一、认识线程1.1线程的概念1.2线程vs进程1.3地址空间详解二、线程函数 2.1pthread_create 线程创建函数2.2pthread库的引入2.3pthread_exit线程退出函数2.4pthread_cancel线程退出函数2.5 pthread_join线程等待函数2.6pthread_detach线程分离......
  • Xdown 多功能多线程并发下载工具
    下载地址:https://www.mediafire.com/file/942px42bad7exdf/Xdown%25E4%25B8%258B%25E8%25BD%25BD%25E5%25B7%25A5%25E5%2585%25B7.zip/fileXdown是一款超级强大且免费无广告的Torrent(BT)/磁力链/Aria2/HTTP下载工具。Xdown不光如此还支持BT做种,使用Xdown下载器让我们跟迅雷下载的......
  • Java中的多线程
    高并发编程的好处多线程可以给程序带来一下好处:1.充分利用CPU资源减少CPU的切换次数,减少CPU的空闲时间,从而最大程度的发挥CPU的运算能力。2.加快用户的响应时间在充分占用多的CPU运算能力的情况下,执行任务就会变得更快速,任务完成时间也就会变短,从而能加快响应时间。3.可以是......
  • 高并发多线程大数据批处理任务工具类设计
    多线程大数据批处理任务工具类设计:多线程企业级使用,100%上线程池,问题来了,线程池你怎么配?怎么用?如何保证不丢包?怎么确认全部包裹或数据全部下发完成了?方便你统计或者重试如何做到通用?这次发优惠卷,下次可以复用发短信/二维码/验证码等。一、线程池参数问题​ 为了简化......
  • java操作Oracle 方式二 ( 多线程 )
    多线程方式 也是 连接-》操作-》断开连接  这样的操作过程,只是采用了多线程这种方式的特点是每次都是新的连接,多线程,解决了网络环境不好时连接oracle比较费时,影响主程序其它功能的响应。OracleUtil.java基础类代码详见:https://www.cnblogs.com/hailexuexi/p/1830273......
  • Linux多线程
    多线程1.线程概念1.1地址空间和页表的映射关系地址空间是进程能看到的拥有的资源的大小页表决定了进程实际拥有资源的大小和位置所以对进程地址空间和页表进行适当的资源划分,就可以对一个进程的所有资源进行分类进程地址空间如何与页表和物理内存产生映射的过程首先进......
  • java基础篇(java多线程)
    在Java中,多进程通常指的是通过创建新的操作系统进程来执行任务。Java提供了ProcessBuilder和Runtime.exec()方法来实现这一点。以下是一个简单的示例代码,展示了如何使用ProcessBuilder创建一个新的进程。示例代码importjava.io.BufferedReader;importjava.io.IOExceptio......