并发编程之 sleep 与 yield的详细解析

news/2024/5/20 10:19:57 标签: java, 算法, juc, 并发编程

3.7 sleep 与 yield

sleep

  1. 调用 sleep 会让当前线程从 Running 进入 Timed Waiting 状态(阻塞)

  2. 其它线程可以使用 interrupt 方法打断正在睡眠的线程,这时 sleep 方法会抛出 InterruptedException

    java">public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread("t1") {
            @Override
            public void run() {
                log.debug("enter sleep...");
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    log.debug("wake up...");
                    e.printStackTrace();
                }
            }
        };
        t1.start();
    ​
        Thread.sleep(1000);
        log.debug("interrupt...");
        t1.interrupt();
    }

    输出结果:

    java">03:47:18.141 c.Test7 [t1] - enter sleep...
    03:47:19.132 c.Test7 [main] - interrupt...
    03:47:19.132 c.Test7 [t1] - wake up...
    java.lang.InterruptedException: sleep interrupted
        at java.lang.Thread.sleep(Native Method)
        at cn.itcast.test.Test7$1.run(Test7.java:14)

  3. 睡眠结束后的线程未必会立刻得到执行

  4. 建议用 TimeUnit 的 sleep 代替 Thread 的 sleep 来获得更好的可读性 。其底层还是sleep方法。

    java">@Slf4j(topic = "c.Test8")
    public class Test8 {
    ​
        public static void main(String[] args) throws InterruptedException {
            log.debug("enter");
            TimeUnit.SECONDS.sleep(1);
            log.debug("end");
    //        Thread.sleep(1000);
        }
    }

  5. 在循环访问锁的过程中,可以加入sleep让线程阻塞时间,防止大量占用cpu资源。

yield

  1. 调用 yield 会让当前线程从 Running 进入 Runnable 就绪状态,然后调度执行其它线程

  2. 具体的实现依赖于操作系统的任务调度器

线程优先级

  • 线程优先级会提示(hint)调度器优先调度该线程,但它仅仅是一个提示,调度器可以忽略它

  • 如果 cpu 比较忙,那么优先级高的线程会获得更多的时间片,但 cpu 闲时,优先级几乎没作用

测试优先级和yield

java">@Slf4j(topic = "c.TestYield")
public class TestYield {
    public static void main(String[] args) {
        Runnable task1 = () -> {
            int count = 0;
            for (;;) {
                System.out.println("---->1 " + count++);
            }
        };
        Runnable task2 = () -> {
            int count = 0;
            for (;;) {
//                Thread.yield();
                System.out.println("              ---->2 " + count++);
            }
        };
        Thread t1 = new Thread(task1, "t1");
        Thread t2 = new Thread(task2, "t2");
        t1.setPriority(Thread.MIN_PRIORITY);
        t2.setPriority(Thread.MAX_PRIORITY);
        t1.start();
        t2.start();
    }
}

测试结果:

java">#优先级
---->1 283500
---->2 374389
#yield
---->1 119199
---->2 101074

可以看出,线程优先级和yield会对线程获取cpu时间片产生一定影响,但不会影响太大。


http://www.niftyadmin.cn/n/5430751.html

相关文章

html--宠物

文章目录 htmljscss html <!DOCTYPE html> <html lang"en" > <head><meta charset"UTF-8"><title>CodePen - Spaceworm</title><script> window.requestAnimFrame (function() {return (window.requestAnimat…

【安全面试宝典】 面试宝典免费分享 面试注意点 附大纲 把握机会 赢得Offer

面试宝典V1.0 仔细阅读 把握机会 赢得Offer 1、 请你做一下自我介绍&#xff1f; 3 2、 你觉得你最大的优点是什么&#xff1f; 3 3、 说说你最大的缺点&#xff1f; 3 4、 你对加班的看法&#xff1f; 3 5、 如何安排自己的时间&#xff1f;会不会排斥加班&#xff1f; 3 6、 …

SpringMVC【框架】

为什么使用Spring ? 1). 方便解耦&#xff0c;简化开发 通过Spring提供的IoC容器&#xff0c;可以将对象之间的依赖关系交由Spring进行控制&#xff0c;避免硬编码所造成的过度程序耦合。 2). AOP编程的支持 通过Spring提供的AOP功能&#xff0c;方便进行面向切面的编程&a…

【Leetcode每日一题】 递归 - 反转链表(难度⭐)(35)

1. 题目解析 题目链接&#xff1a;206. 反转链表 这个问题的理解其实相当简单&#xff0c;只需看一下示例&#xff0c;基本就能明白其含义了。 2.算法原理 一、递归函数的核心任务 递归函数的主要职责是接受一个链表的头指针&#xff0c;并返回该链表逆序后的新头结点。递归…

红队基础建设与介绍

1.ATT&CK相关背景 ATT&CK在各种日常环境中都很有价值。开展任何防御活动时&#xff0c;可以应用ATT&CK防御法&#xff0c;参考攻击者及其行为。ATT&CK不仅对网络防御者提供通用技术库&#xff0c;还为渗透测试和红队提供了基础。提到对抗行为时&#xff0c;这为…

SpringBoot+vue3打造企业级一体化SaaS系统

SpringBootvue3打造企业级一体化SaaS系统 简介&#xff1a;    全面提升前后端技术水平&#xff0c;独立完成全栈项目开发能力&#xff0c;快速进击全栈工程师&#xff0c;最终在面试中脱颖而出。整合后端主流技术&#xff08;Spring Boot、物理数据库隔离、加载动态权限、多…

JAVA 100道题目(1)

1.编写一个程序&#xff0c;输入两个整数并计算它们的和 以下是一个简单的Java程序&#xff0c;用于输入两个整数并计算它们的和&#xff1a; java复制代码 import java.util.Scanner; public class IntegerSum { public static void main(String[] args) { Scanner scanner …