Java线程优先级

Java 线程优先级

Thread 类中,使用如下属性来代表优先级。

1privateint priority;

我们可以通过 setPriority(int newPriority) 来设置新的优先级,通过 getPriority() 来获取线程的优先级。

有些资料通过下面的例子就得出了一个结论:Java 线程默认优先级是 5。

publicstaticvoid main(String[] args) {

    Thread thread = newThread();

    System.out.println(thread.getPriority());

}

// 打印结果:5

其实这是大错特错的,只是看到了表面,看看下面的例子,我们把当前线程的优先级改为 4,发现子线程 thread 的优先级也是 4。

publicstaticvoid main(String[] args) {

    Thread.currentThread().setPriority(4);

    Thread thread = newThread();

    System.out.println(thread.getPriority());

}


// 打印结果:4

这啪啪啪打脸了,如果是线程默认优先级是 5,我们新创建的 thread 线程,没设置优先级,理应是 5,但实际是 4。我们看看 Thread 初始化 priority 的源代码。

Thread parent = currentThread();

this.priority = parent.getPriority();

原来,线程默认的优先级是继承父线程的优先级,上面例子我们把父线程的优先级设置为 4,所以导致子线程的优先级也变成 4。

严谨一点说,子线程默认优先级和父线程一样,Java 主线程默认的优先级是 5。

Java 中定义了 3 种优先级,分别是最低优先级(1)、正常优先级(5)、最高优先级(10),代码如下所示。Java 优先级范围是 [1, 10],设置其他数字的优先级都会抛出 IllegalArgumentException 异常。

/**

 * The minimum priority that a thread can have.

 */

publicfinalstaticint MIN_PRIORITY = 1;


/**

 * The default priority that is assigned to a thread.

 */

publicfinalstaticint NORM_PRIORITY = 5;


/**

 * The maximum priority that a thread can have.

 */

publicfinalstaticint MAX_PRIORITY = 10;

接下来说说线程优先级的作用。先看下面代码,代码逻辑是创建了 3000 个线程,分别是: 1000 个优先级为 1 的线程, 1000 个优先级为 5 的线程,1000 个优先级为 10 的线程。用 minTimes 来记录 1000 个 MIN_PRIORITY 线程运行时时间戳之和,用 normTimes 来记录 1000 个 NORM_PRIORITY 线程运行时时间戳之和,用 maxTimes 来记录 1000 个 MAX_PRIORITY 线程运行时时间戳之和。通过统计每个优先级的运行的时间戳之和,值越小代表的就是越优先执行。我们运行看看。

publicclassTestPriority {

    staticAtomicLong minTimes = newAtomicLong(0);

    staticAtomicLong normTimes = newAtomicLong(0);

    staticAtomicLong maxTimes = newAtomicLong(0);


    publicstaticvoid main(String[] args) {

        List<MyThread> minThreadList = newArrayList<>();

        List<MyThread> normThreadList = newArrayList<>();

        List<MyThread> maxThreadList = newArrayList<>();


        int count= 1000;

        for(int i = 0; i < count; i++) {

            MyThread myThread = newMyThread("min----"+ i);

            myThread.setPriority(Thread.MIN_PRIORITY);

            minThreadList.add(myThread);

        }

        for(int i = 0; i < count; i++) {

            MyThread myThread = newMyThread("norm---"+ i);

            myThread.setPriority(Thread.NORM_PRIORITY);

            normThreadList.add(myThread);

        }

        for(int i = 0; i < count; i++) {

            MyThread myThread = newMyThread("max----"+ i);

            myThread.setPriority(Thread.MAX_PRIORITY);

            maxThreadList.add(myThread);

        }


        for(int i = 0; i < count; i++) {

            maxThreadList.get(i).start();

            normThreadList.get(i).start();

            minThreadList.get(i).start();

        }


        try{

            Thread.sleep(3000);

        } catch(InterruptedException e) {

            e.printStackTrace();

        }


        System.out.println("maxPriority 统计:"+ maxTimes.get());

        System.out.println("normPriority 统计:"+ normTimes.get());

        System.out.println("minPriority 统计:"+ minTimes.get());

        System.out.println("普通优先级与最高优先级相差时间:"+ (normTimes.get() - maxTimes.get()) + "ms");

        System.out.println("最低优先级与普通优先级相差时间:"+ (minTimes.get() - normTimes.get()) + "ms");


    }


    staticclassMyThread extendsThread {


        publicMyThread(String name) {

            super(name);

        }


        @Override

        publicvoid run() {

            System.out.println(this.getName() + " priority: "+ this.getPriority());

            switch(this.getPriority()) {

                caseThread.MAX_PRIORITY :

                    maxTimes.getAndAdd(System.currentTimeMillis());

                    break;

                caseThread.NORM_PRIORITY :

                    normTimes.getAndAdd(System.currentTimeMillis());

                    break;

                caseThread.MIN_PRIORITY :

                    minTimes.getAndAdd(System.currentTimeMillis());

                    break;

                default:

                    break;

            }

        }

    }

}

执行结果如下:

# 第一部分

max----0 priority: 10

norm---0 priority: 5

max----1 priority: 10

max----2 priority: 10

norm---2 priority: 5

min----4 priority: 1

.......

max----899 priority: 10

min----912 priority: 1

min----847 priority: 5

min----883 priority: 1


# 第二部分

maxPriority 统计:1568986695523243

normPriority 统计:1568986695526080

minPriority 统计:1568986695545414

普通优先级与最高优先级相差时间:2837ms

最低优先级与普通优先级相差时间:19334ms

我们一起来分析一下结果。先看看第一部分,最开始执行的线程高优先级、普通优先级、低优先级都有,最后执行的线程也都有各个优先级的,这说明了:优先级高的线程不代表一定比优先级低的线程优先执行。也可以换另一种说法:代码执行顺序跟线程的优先级无关。看看第二部分的结果,我们可以发现最高优先级的 1000 个线程执行时间戳之和最小,而最低优先级的 1000 个线程执行时间戳之和最大,因此可以得知:一批高优先级的线程会比一批低优先级的线程优先执行,即高优先级的线程大概率比低优先的线程优先获得 CPU 资源。

各操作系统中真有 10 个线程等级么?

Java 作为跨平台语言,线程有 10 个等级,但是映射到不同操作系统的线程优先级值不一样。接下来教大家怎么在 OpenJDK 源码中查各个操作系统中线程优先级映射的值。

看到 Thread 源代码,设置线程优先级最终调用了本地方法 setPriority0();

privatenative void setPriority0(int newPriority);

接着我们在 OpenJDK 的 Thread.c 代码中找到 setPriority0() 对应的方法 JVM_SetThreadPriority;

staticJNINativeMethod methods[] = {

    ...

    {"setPriority0",     "(I)V",       (void *)&JVM_SetThreadPriority},

    ...};

我们根据 JVM_SetThreadPriority 找到 jvm.cpp 中对应的代码段;

JVM_ENTRY(void, JVM_SetThreadPriority(JNIEnv* env, jobject jthread, jint prio))

  JVMWrapper("JVM_SetThreadPriority");

  // Ensure that the C++ Thread and OSThread structures aren't freed before we operate

  MutexLocker ml(Threads_lock);

  oop java_thread = JNIHandles::resolve_non_null(jthread);

  java_lang_Thread::set_priority(java_thread, (ThreadPriority)prio);

  JavaThread* thr = java_lang_Thread::thread(java_thread);

  if(thr != NULL) {                  // Thread not yet started; priority pushed down when it is

    Thread::set_priority(thr, (ThreadPriority)prio);

  }

JVM_END

根据第 3 步中的代码,我们可以发现关键是 java_lang_Thread::set_Priority() 这段代码,继续找 thread.cpp 代码中的 set_Priority() 方法;

void Thread::set_priority(Thread* thread, ThreadPriority priority) {

  trace("set priority", thread);

  debug_only(check_for_dangling_thread_pointer(thread);)

  // Can return an error!

  (void)os::set_priority(thread, priority);

}

本文来自PHP中文网的java学习教程栏目:https://www.php.cn/java/

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,602评论 1 15
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 3,115评论 1 18
  • 文章来源:http://www.54tianzhisheng.cn/2017/06/04/Java-Thread/...
    beneke阅读 1,927评论 0 1
  • 在讨论多线程之前,让我们先讨论线程。线程是进程中轻量级的最小部分,可以与同一进程的其他部分(其他线程)并发运行。线...
    墨雨轩夏阅读 207评论 0 0
  • 栀子花开 so beautiful so white 在这个城市第一次养花,这盛开的花朵让我忍不住想拍个照分享一下...
    天涯星辰阅读 365评论 0 0

友情链接更多精彩内容