Java中的线程让步会让线程让出优先级,而休眠则会让线程进入阻塞状态等待被唤醒,这里我们对比线程等待的wait()方法,来详解Java中的线程让步yield()与线程休眠sleep()方法

线程让步: yield()yield()的作用是让步。它能让当前线程由“运行状态”进入到“就绪状态”,从而让其它具有相同优先级的等待线程获取执行权;但是,并不能保证在当前线程调用yield()之后,其它具有相同优先级的线程就一定能获得执行权;也有可能是当前线程又进入到“运行状态”继续运行!

示例:

class ThreadA extends Thread{

public ThreadA(String name){

super(name);

}

public synchronized void run(){

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

System.out.printf("%s [%d]:%d\n", this.getName(), this.getPriority(), i);

// i整除4时,调用yield

if (i%4 == 0)

Thread.yield();

}

}

}

public class YieldTest{

public static void main(String[] args){

ThreadA t1 = new ThreadA("t1");

ThreadA t2 = new ThreadA("t2");

t1.start();

t2.start();

}

}

(某一次的)运行结果:

t1 [5]:0

t2 [5]:0

t1 [5]:1

t1 [5]:2

t1 [5]:3

t1 [5]:4

t1 [5]:5

t1 [5]:6

t1 [5]:7

t1 [5]:8

t1 [5]:9

t2 [5]:1

t2 [5]:2

t2 [5]:3

t2 [5]:4

t2 [5]:5

t2 [5]:6

t2 [5]:7

t2 [5]:8

t2 [5]:9

结果说明:

“线程t1”在能被4整数的时候,并没有切换到“线程t2”。这表明,yield()虽然可以让线程由“运行状态”进入到“就绪状态”;但是,它不一定会让其它线程获取CPU执行权(即,其它线程进入到“运行状态”),即使这个“其它线程”与当前调用yield()的线程具有相同的优先级。

yield() 与 wait()的比较:我们知道,wait()的作用是让当前线程由“运行状态”进入“等待(阻塞)状态”的同时,也会释放同步锁。而yield()的作用是让步,它也会让当前线程离开“运行状态”。它们的区别是:

(1) wait()是让线程由“运行状态”进入到“等待(阻塞)状态”,而不yield()是让线程由“运行状态”进入到“就绪状态”。

(2) wait()是会线程释放它所持有对象的同步锁,而yield()方法不会释放锁。

下面通过示例演示yield()是不会释放锁的:

public class YieldLockTest{

private static Object obj = new Object();

public static void main(String[] args){

ThreadA t1 = new ThreadA("t1");

ThreadA t2 = new ThreadA("t2");

t1.start();

t2.start();

}

static class ThreadA extends Thread{

public ThreadA(String name){

super(name);

}

public void run(){

// 获取obj对象的同步锁

synchronized (obj) {

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

System.out.printf("%s [%d]:%d\n", this.getName(), this.getPriority(), i);

// i整除4时,调用yield

if (i%4 == 0)

Thread.yield();

}

}

}

}

}

(某一次)运行结果:

t1 [5]:0

t1 [5]:1

t1 [5]:2

t1 [5]:3

t1 [5]:4

t1 [5]:5

t1 [5]:6

t1 [5]:7

t1 [5]:8

t1 [5]:9

t2 [5]:0

t2 [5]:1

t2 [5]:2

t2 [5]:3

t2 [5]:4

t2 [5]:5

t2 [5]:6

t2 [5]:7

t2 [5]:8

t2 [5]:9

结果说明:

主线程main中启动了两个线程t1和t2。t1和t2在run()会引用同一个对象的同步锁,即synchronized(obj)。在t1运行过程中,虽然它会调用Thread.yield();但是,t2是不会获取cpu执行权的。因为,t1并没有释放“obj所持有的同步锁”!

线程休眠:sleep()sleep() 定义在Thread.java中。

sleep() 的作用是让当前线程休眠,即当前线程会从“运行状态”进入到“休眠(阻塞)状态”。sleep()会指定休眠时间,线程休眠的时间会大于/等于该休眠时间;在线程重新被唤醒时,它会由“阻塞状态”变成“就绪状态”,从而等待cpu的调度执行。

示例:

class ThreadA extends Thread{

public ThreadA(String name){

super(name);

}

public synchronized void run() {

try {

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

System.out.printf("%s: %d\n", this.getName(), i);

// i能被4整除时,休眠100毫秒

if (i%4 == 0)

Thread.sleep(100);

}

} catch (InterruptedException e) {

e.printStackTrace();

}

}

}

public class SleepTest{

public static void main(String[] args){

ThreadA t1 = new ThreadA("t1");

t1.start();

}

}

运行结果:

t1: 0

t1: 1

t1: 2

t1: 3

t1: 4

t1: 5

t1: 6

t1: 7

t1: 8

t1: 9

结果说明:

程序比较简单,在主线程main中启动线程t1。t1启动之后,当t1中的计算i能被4整除时,t1会通过Thread.sleep(100)休眠100毫秒。

sleep() 与 wait()的比较:我们知道,wait()的作用是让当前线程由“运行状态”进入“等待(阻塞)状态”的同时,也会释放同步锁。而sleep()的作用是也是让当前线程由“运行状态”进入到“休眠(阻塞)状态”。

但是,wait()会释放对象的同步锁,而sleep()则不会释放锁。

下面通过示例演示sleep()是不会释放锁的。

public class SleepLockTest{

private static Object obj = new Object();

public static void main(String[] args){

ThreadA t1 = new ThreadA("t1");

ThreadA t2 = new ThreadA("t2");

t1.start();

t2.start();

}

static class ThreadA extends Thread{

public ThreadA(String name){

super(name);

}

public void run(){

// 获取obj对象的同步锁

synchronized (obj) {

try {

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

System.out.printf("%s: %d\n", this.getName(), i);

// i能被4整除时,休眠100毫秒

if (i%4 == 0)

Thread.sleep(100);

}

} catch (InterruptedException e) {

e.printStackTrace();

}

}

}

}

}

运行结果:

t1: 0

t1: 1

t1: 2

t1: 3

t1: 4

t1: 5

t1: 6

t1: 7

t1: 8

t1: 9

t2: 0

t2: 1

t2: 2

t2: 3

t2: 4

t2: 5

t2: 6

t2: 7

t2: 8

t2: 9

结果说明:

主线程main中启动了两个线程t1和t2。t1和t2在run()会引用同一个对象的同步锁,即synchronized(obj)。在t1运行过程中,虽然它会调用Thread.sleep(100);但是,t2是不会获取cpu执行权的。因为,t1并没有释放“obj所持有的同步锁”!

注意,若我们注释掉synchronized (obj)后再次执行该程序,t1和t2是可以相互切换的。下面是注释调synchronized(obj) 之后的源码:

public class SleepLockTest{

private static Object obj = new Object();

public static void main(String[] args){

ThreadA t1 = new ThreadA("t1");

ThreadA t2 = new ThreadA("t2");

t1.start();

t2.start();

}

static class ThreadA extends Thread{

public ThreadA(String name){

super(name);

}

public void run(){

// 获取obj对象的同步锁

// synchronized (obj) {

try {

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

System.out.printf("%s: %d\n", this.getName(), i);

// i能被4整除时,休眠100毫秒

if (i%4 == 0)

Thread.sleep(100);

}

} catch (InterruptedException e) {

e.printStackTrace();

}

// }

}

}

}

java sleep唤醒_详解Java中的线程让步yield()与线程休眠sleep()方法相关推荐

  1. java comparator相等_详解Java中Comparable和Comparator接口的区别

    详解Java中Comparable和Comparator接口的区别 发布于 2020-7-20| 复制链接 摘记: 详解Java中Comparable和Comparator接口的区别本文要来详细分析一 ...

  2. Java implement意思_详解JAVA中implement和extends的区别

    详解JAVA中implement和extends的区别 发布于 2020-4-14| 复制链接 摘记: 详解JAVA中implement和extends的区别extends是继承父类,只要那个类不是声 ...

  3. java runnable 异常_详解Java中多线程异常捕获Runnable的实现

    详解Java中多线程异常捕获Runnable的实现 1.背景: Java 多线程异常不向主线程抛,自己处理,外部捕获不了异常.所以要实现主线程对子线程异常的捕获. 2.工具: 实现Runnable接口 ...

  4. java 代码块_详解java中的四种代码块

    在java中用{}括起来的称为代码块,代码块可分为以下四种: 一.简介 1.普通代码块: 类中方法的方法体 2.构造代码块: 构造块会在创建对象时被调用,每次创建时都会被调用,优先于类构造函数执行. ...

  5. java static 函数_详解java中的static关键字

    Java中的static关键字可以用于修饰变量.方法.代码块和类,还可以与import关键字联合使用,使用的方式不同赋予了static关键字不同的作用,且在开发中使用广泛,这里做一下深入了解. 静态资 ...

  6. java 标量替换_详解jvm中的标量替换

    概述 通常在java中创建一个对象,大家都认为是在堆中创建. 在jdk6开始有逃逸分析,标量替换等技术,关于在堆中创建对象不再绝对. 关于标量替换,通过以下几点进行概述: 逃逸分析 标量替换是什么 测 ...

  7. java 引用传递_详解java的值传递、地址传递、引用传递

    详解java的值传递.地址传递.引用传递 一直来觉得对值传递和地址传递了解的很清楚,刚才在开源中国上看到一篇帖子介绍了java中的值传递和地址传递,看完后感受颇深.下边总结下以便更容易理解. 按照以前 ...

  8. java内部格式_详解java内部类的访问格式和规则

    详解java内部类的访问格式和规则 1.内部类的定义 定义一个类来描述事物,但是这个事物其中可能还有事物,这时候在类中再定义类来描述. 2.内部类访问规则 ①内部类可以直接访问外部类中的成员,包括私有 ...

  9. python java混合编程_详解java调用python的几种用法(看这篇就够了)

    java调用python的几种用法如下: 在java类中直接执行python语句 在java类中直接调用本地python脚本 使用Runtime.getRuntime()执行python脚本文件(推荐 ...

最新文章

  1. java输入字符串异常_设计一个 Java 程序,自定义异常类,从命令行(键盘)输入一个字符串,如果该字符串值为“XYZ”。。。...
  2. 通过读源码win10驱动下实现3环的GetEnvironmentVariable
  3. 46亿一辆的戴森电动车,「卖」给英国首富了
  4. python资源百度网盘-bypy-百度网盘Python客户端 linux
  5. BLE控制器之物理层特性
  6. hbase操作,创建命名空间,创建hbase表,添加数据,删除数据,通过RowKey获取数据,扫描数据,获取所有的表,获取ColumnFamily,删除表,获取hbase命名空间
  7. 高斯滤波和双向滤波的区别与联系
  8. 华为3COM交换机PVLAN配置使用说明
  9. 11.python之线程,协程,进程,
  10. 微信获取地理位置转城市demo
  11. Logstash自定义grok正则匹配规则
  12. 计算机图学与工业设计,工业设计和工程制图的关系
  13. Android APP 登陆界面开发 UI设计
  14. 无法启动此程序因为计算机中丢失adapt,【完美解决】Wi7中NVIDIA GT540M安装程序无法找到和你现有硬件兼容的任何驱动程序,安装程序将会退出.doc...
  15. python海龟交易策略_python 海龟交易法则 股票回测-双均线规则(一)
  16. oracle创建完成 sys密码怎么修改,如何重置密码 oracle sys和system
  17. perl中unicode属性
  18. 载入pytorch的预训练模型时遇到_pickle.UnpicklingError: unpickling stack underflow
  19. Postman导入数据批量测试
  20. 地壳中元素含量排名记忆口诀_地壳中元素含量_地壳中元素含量口诀

热门文章

  1. 2024武汉理工大学计算机考研信息汇总
  2. Flink/Hbase 异常 - 4.Sink 背压100% 与 hbase.util.RetryCounter.sleepUntilNextRetry 异常分析与排查
  3. 实现全选或者多选删除
  4. python怎么计算_python怎么计算
  5. shogun-toolbox的安装与问题总结
  6. Java程序员花十个月做私活,收入50w+,网友:老哥建个群搭个伙呗
  7. 苹果字体对应font-weight大小
  8. 八皇后问题动态演示_Qt5实现
  9. TortoiseSVN文件图标大全
  10. HDU 4262 Juggler (模拟+线段树优化)