【JavaSE专栏84】线程让步,一种线程调度的机制

简介: 【JavaSE专栏84】线程让步,一种线程调度的机制

本文讲解了 Java 中线程让步的语法和应用场景,并给出了样例代码。线程让步是一种线程调度的机制,当线程调用 Thread.yield() 方法时,它就会主动放弃当前的CPU执行时间片,让给其他具有相同优先级的线程执行。


一、什么是线程让步

线程让步是一种线程调度的机制

当线程调用 Thread.yield() 方法时,它就会主动放弃当前的CPU执行时间片,让给其他具有相同优先级的线程执行。

实际上,线程让步是一种提醒调度器当前线程已经完成了当前阶段的任务,可以进行线程切换,但是不保证一定发生线程切换,调度器可能会忽略这个提醒,继续执行当前线程。

调用 Thread.yield() 的线程会从运行状态转换到就绪状态,与其他就绪状态的线程竞争 CPU 的执行权。

如果没有其他线程处于就绪状态,那么当前线程仍然可以继续执行。

但是,一般情况下,调用 Thread.yield() 会导致当前线程让出 CPU 的使用权,使得其他具有相同优先级的线程有机会获得执行。

线程让步的主要作用是使得多个具有相同优先级的线程能够公平竞争 CPU 的执行时间,避免某个线程长时间霸占 CPU 而导致其他线程无法执行的情况,但线程让步并不保证公平性,具体的执行顺序还是由操作系统的线程调度器决定。

过度使用线程让步可能会导致性能下降,因为频繁地进行线程切换会带来一定的开销,因此开发者应该谨慎使用线程让步,只在必要的时候使用。


二、什么情况下产生线程让步

线程让步是一种线程调度机制,线程可以主动放弃当前的CPU执行时间片,让给其他具有相同优先级的线程执行。

线程让步的情况主要有以下 3 33 种,请同学们认真学习。

  1. 当前线程执行完了一部分任务,希望让其他线程有机会执行。可以通过调用 Thread 类的 yield() 方法来实现线程让步。调用 yield() 方法会让当前线程从运行状态转换到就绪状态,与其他具有相同优先级的线程竞争 CPU 的执行权。
  2. 当前线程遇到一些临时的阻塞条件,希望让其他线程先执行。例如,线程在执行过程中发现某个共享资源不可用,为了避免浪费 CPU 的时间,可以让出执行权,等待资源可用后再继续执行。
  3. 当前线程执行的任务比较重,为了避免长时间占用 CPU 而导致其他线程无法执行,可以适当让步,让其他线程有机会执行。

线程让步并不保证一定会发生线程切换,具体是否发生线程切换还是由操作系统的线程调度器决定,另外过度地使用线程让步可能会导致性能下降,因为频繁地进行线程切换会带来一定的开销,所以在使用线程让步时需要谨慎考虑。


三、模拟线程让步

以下是一个使用 Java 模拟线程让步的示例代码,请同学们复制到本地执行。

public class ThreadYieldExample {
    public static void main(String[] args) {
        Runnable myRunnable = new MyRunnable();
        Thread thread1 = new Thread(myRunnable, "Thread 1");
        Thread thread2 = new Thread(myRunnable, "Thread 2");
        thread1.start();
        thread2.start();
    }
    static class MyRunnable implements Runnable {
        @Override
        public void run() {
            for (int i = 0; i < 5; i++) {
                System.out.println(Thread.currentThread().getName() + " is running");
                // 模拟某个条件需要让步,让其他线程先执行
                if (i == 2 && Thread.currentThread().getName().equals("Thread 1")) {
                    System.out.println(Thread.currentThread().getName() + " is yielding");
                    Thread.yield();
                }
                try {
                    Thread.sleep(500);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

在这个示例中,我们创建了两个线程 Thread 1Thread 2,它们都执行同一个 MyRunnable 任务。

MyRunnablerun() 方法中,每个线程会打印自己的名字并休眠 500 500500 毫秒。

i 等于 2 22 且当前线程的名字是 Thread 1 时,我们通过 Thread.yield() 方法进行线程让步,让给其他线程执行。

运行程序后,可以看到在 Thread 1 执行到一定条件时,它会让步给 Thread 2 执行一段时间,然后继续自己的执行。

这样就模拟了 Java 线程让步的效果。


四、线程让步的应用场景

线程让步是一种线程调度机制,可以让当前线程主动放弃 CPU 执行时间片,让给其他具有相同优先级的线程执行。

线程让步的应用场景主要包括以下 4 44 种情况,请同学们认真学习。

  1. 公平竞争 CPU 资源:在多线程环境下,希望各个线程能够公平地竞争 CPU 的执行时间,避免某个线程长时间占用 CPU 而导致其他线程无法执行。通过在适当的时候调用 Thread.yield() 方法,可以让当前线程让出 CPU 执行权,给其他线程执行的机会。
  2. 协同处理任务:在某些任务中,不同线程的处理速度可能不一致,为了提高整体的处理效率,可以让处理速度较快的线程让步,让处理速度较慢的线程有机会追赶上来。这样可以确保各个线程以合理的速度协同工作。
  3. 避免线程饥饿:线程饥饿是指某个线程由于优先级较低或其他原因,一直无法获取到 CPU 的执行时间片,导致无法执行任务。通过适当使用线程让步,可以让优先级较低的线程有机会执行,避免线程饥饿的问题。
  4. 提高响应性:在某些场景下,需要实现较为及时的响应,例如处理用户请求、消息处理等,通过在关键代码片段中适当调用 Thread.yield() 方法,可以让其他线程有机会及时执行,提高整体的响应性能。

在使用线程让步时需要谨慎,过度使用可能会导致性能下降,合理地使用线程让步可以提升多线程程序的效率和响应性。


五、线程让步面试题

问题:什么是线程让步?如何在Java中实现线程让步?

线程让步是一种线程调度机制,它允许一个线程主动放弃 CPU 执行时间片,让给其他具有相同优先级的线程执行。

在 Java 中,可以通过 Thread 类的 yield() 方法来实现线程让步。

调用 yield() 方法会使当前线程从运行状态转换到就绪状态,然后与其他具有相同优先级的线程竞争 CPU 的执行权。

调用 yield() 方法的线程并不能保证一定会切换到其他线程,具体是否发生线程切换还是由操作系统的线程调度器决定。

示例代码如下。

public class ThreadYieldExample {
    public static void main(String[] args) {
        Runnable myRunnable = new MyRunnable();
        Thread thread1 = new Thread(myRunnable, "Thread1");
        Thread thread2 = new Thread(myRunnable, "Thread2");
        thread1.start();
        thread2.start();
    }
    static class MyRunnable implements Runnable {
        @Override
        public void run() {
            for (int i = 0; i < 5; i++) {
                System.out.println(Thread.currentThread().getName() + " is running");
                // 模拟某个条件需要让步,让其他线程先执行
                if (i == 2 && Thread.currentThread().getName().equals("Thread1")) {
                    System.out.println(Thread.currentThread().getName() + " is yielding");
                    Thread.yield();
                }
                try {
                    Thread.sleep(500);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

六、总结

本文讲解了 Java 中线程让步的语法和应用场景,并给出了样例代码,在下一篇博客中,将讲解 Java 线程优先权的问题。


相关文章
|
2月前
|
Java API 调度
线程的中断(interrupt)机制
线程的中断(interrupt)机制
30 1
|
2月前
|
人工智能 JSON 前端开发
【Spring boot实战】Springboot+对话ai模型整体框架+高并发线程机制处理优化+提示词工程效果展示(按照框架自己修改可对接市面上百分之99的模型)
【Spring boot实战】Springboot+对话ai模型整体框架+高并发线程机制处理优化+提示词工程效果展示(按照框架自己修改可对接市面上百分之99的模型)
|
24天前
|
监控 Java
解析Java线程池的异常处理机制
该内容是一个关于Java线程和线程池异常处理的总结。提到的关键点包括: 1. 引用了滑动验证页面和相关文章资源。 2. 区分了`execute`与`submit`在处理线程异常时的区别,`submit`可能会捕获并隐藏异常,而`execute`会直接抛出。 3. 提供了处理线程和线程池异常的建议,如使用try/catch直接捕获,或者自定义线程工厂和未捕获异常处理器。 4. 示例代码展示了如何通过设置`UncaughtExceptionHandler`来监控和处理线程中的异常。 请注意,由于字符限制,这里只提供了简要摘要,详细解释和代码示例请参考原文。
22 3
|
2月前
|
资源调度 算法 Linux
Linux进程/线程的调度机制介绍:详细解析Linux系统中进程/线程的调度优先级规则
Linux进程/线程的调度机制介绍:详细解析Linux系统中进程/线程的调度优先级规则
118 0
|
1天前
|
安全 算法 关系型数据库
线程安全--深入探究线程等待机制和死锁问题
线程安全--深入探究线程等待机制和死锁问题
|
6天前
|
安全 算法 Java
JavaSE&多线程&线程池
JavaSE&多线程&线程池
127 7
|
11天前
|
安全 Java 编译器
【JavaEE多线程】线程安全、锁机制及线程间通信
【JavaEE多线程】线程安全、锁机制及线程间通信
31 1
|
14天前
|
监控 Java 关系型数据库
JVM工作原理与实战(十三):打破双亲委派机制-线程上下文类加载器
JVM作为Java程序的运行环境,其负责解释和执行字节码,管理内存,确保安全,支持多线程和提供性能监控工具,以及确保程序的跨平台运行。本文主要介绍了打破双亲委派机制的方法、线程上下文类加载器等内容。
14 2
|
24天前
|
安全 Java 调度
深入理解Java中的线程安全与锁机制
【4月更文挑战第6天】 在并发编程领域,Java语言提供了强大的线程支持和同步机制来确保多线程环境下的数据一致性和线程安全性。本文将深入探讨Java中线程安全的概念、常见的线程安全问题以及如何使用不同的锁机制来解决这些问题。我们将从基本的synchronized关键字开始,到显式锁(如ReentrantLock),再到读写锁(ReadWriteLock)的讨论,并结合实例代码来展示它们在实际开发中的应用。通过本文,读者不仅能够理解线程安全的重要性,还能掌握如何有效地在Java中应用各种锁机制以保障程序的稳定运行。
|
2月前
|
算法 Linux 调度
Linux 线程介绍:介绍Linux系统中线程的基本概念、创建和调度机制
Linux 线程介绍:介绍Linux系统中线程的基本概念、创建和调度机制
26 0