当前位置: 首页 > news >正文

【Java深入学习】并发常见方法的注意事项

start 与 run

我们知道 start方法是运行Thread里的run方法,那么我们之间调用run方法,这两者之前的区别是什么

代码示例

import lombok.extern.slf4j.Slf4j;

@Slf4j(topic = "c.Test")
public class Test {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread("t1"){
            @Override
            public void run() {
                log.debug(Thread.currentThread().getName());
            }
        };
        t1.start();
        t1.sleep(100);
        t1.run();
    }
}
复制代码

结果:

  • 18:27:56.487 c.Test [t1] - t1
  • 18:27:56.487 c.Test [main] - main

可以看出 直接调用 run 是在主线程中执行了 run,没有启动新的线程 使用 start 是启动新的线程,通过新的线程间接执行 run 中的代码

sleep 与 yield

两者的区别

sleep

  1. 调用 sleep 会让当前线程从 Running 进入 Timed Waiting 状态(阻塞)
  2. 其它线程可以使用 interrupt 方法打断正在睡眠的线程,这时 sleep 方法会抛出 InterruptedException
  3. 睡眠结束后的线程未必会立刻得到执行
  4. 建议用 TimeUnit 的 sleep 代替 Thread 的 sleep 来获得更好的可读性

yield

  1. 调用 yield 会让当前线程从 Running 进入 Runnable 就绪状态,然后调度执行其它线程
  2. 具体的实现依赖于操作系统的任务调度器

sleep状态

import lombok.extern.slf4j.Slf4j;

@Slf4j(topic = "c.Test6")
public class Test6 {

    public static void main(String[] args) {
        Thread t1 = new Thread("t1") {
            @Override
            public void run() {
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };

        t1.start();
        log.debug("t1 state: {}", t1.getState());

        try {
            Thread.sleep(500);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        log.debug("t1 state: {}", t1.getState());
    }
}
复制代码

结果:

18:53:22.128 c.Test6 [main] - t1 state: RUNNABLE

18:53:22.633 c.Test6 [main] - t1 state: TIMED_WAITING

刚开始调用的时候t1状态为运行态,之后当t1线程调用sleep时 主线程查看t1的状态为TIMED_WAITING(阻塞)

yield状态

import lombok.extern.slf4j.Slf4j;

@Slf4j(topic = "c.TestYield")
public class TestYield {
    public static void main(String[] args) {
        Runnable task1 = () -> {
            System.out.println("---->1 ");
        };
        Runnable task2 = () -> {
            try {
                Thread.sleep(500);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            Thread.yield();
            System.out.println("              ---->2 ");

        };
        Thread t1 = new Thread(task1, "t1");
        Thread t2 = new Thread(task2, "t2");
        t1.start();
        log.debug("t2 state: {}", t2.getState());
        t2.start();
        log.debug("t2 state: {}", t2.getState());

    }

}
复制代码

结果:

---->1

00:28:50.450 c.TestYield [main] - t2 state: NEW

00:28:50.454 c.TestYield [main] - t2 state: RUNNABLE

---->2

可以看出t2线程yield让步后就变为了RUNNABLE就绪态

补充

yield并不是一定会让步,它的原理是把自己让出 然后变成就绪态 然后和其它线程再争抢cpu,所以有可能yield后还是此线程。

代码示例

import lombok.extern.slf4j.Slf4j;

@Slf4j(topic = "c.Test9")
public class Test9 {

    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.start();
        t2.start();
    }
}
复制代码

结果:

  • ---->1 16193
  • ---->1 16194
  • ---->1 16195
  • ---->1 16196
  • ---->1 16197
  • ---->1 16198
  • ---->1 16199
  • ---->1 16200
  • ---->1 16201
  • ---->1 16202
  • ---->1 16203
  • ---->1 16204
  • ---->1 16205
  • ---->1 16206
  • ---->1 16207
  • ---->1 16208
  • ---->1 16209
  • ---->1 16210
  • ---->1 16211
  • ---->2 4458
  • ---->2 4459
  • ---->2 4460
  • ---->2 4461
  • ---->2 4462
  • ---->2 4463
  • ---->2 4464
  • ---->2 4465
  • ---->2 4466
  • ---->2 4467

可以看出yield让步大概率是 让步到其它线程,但并不代表此线程就一定会让步。

再补充

线程优先级会提示调度器优先调度该线程,但它仅仅是一个提示,调度器可以忽略它 如果 cpu 比较忙,那么优先级高的线程会获得更多的时间片,但 cpu 闲时,优先级几乎没作用。可以说它和yield一样 都是不确定是否一定会改变当前线程。

可以看出虽然t2的优先级为MAX_PRIORITY(优先级为10) t1的优先级为MIN_PRIORITY(优先级为1),但是仍然会出现t1不断输出的情况,说明了 优先级只是作为参考 真正决定执行什么线程的是cpu。

相关文章:

  • 微信小程序开发入门与实战(数据监听)
  • 【论文阅读】提升的自动作文评分通过Prompt预测和匹配
  • JPA-Specification常用条件查询构造方式
  • 瑞吉外卖(19) - 新增套餐业务开发
  • Android 的定位分层架构
  • 基于docker搭建es集群
  • 2.可视化基础(上)
  • 解决Vue项目中ESLint和Prettier冲突问题[Vue.js项目实践: 新冠自检系统]
  • 【misc】buu-面具下的flag——zip伪加密+用NTFS流隐藏文件
  • java毕业设计企业安全与设备管理系统源码+lw文档+mybatis+系统+mysql数据库+调试
  • Message Bus Solace Deeper Dive
  • Android-AGP之手写你的第一款自定义plugin插件
  • 入门力扣自学笔记159 C++ (题目编号788)
  • java毕业设计奇妙店铺电子商务网站源码+lw文档+mybatis+系统+mysql数据库+调试
  • 类与对象(十四)----包package
  • 30天自制操作系统-2
  • - C#编程大幅提高OUTLOOK的邮件搜索能力!
  • Github访问慢解决办法
  • JavaScript工作原理(五):深入了解WebSockets,HTTP/2和SSE,以及如何选择
  • MYSQL 的 IF 函数
  • Node项目之评分系统(二)- 数据库设计
  • springMvc学习笔记(2)
  • windows下使用nginx调试简介
  • yii2权限控制rbac之rule详细讲解
  • 大整数乘法-表格法
  • 警报:线上事故之CountDownLatch的威力
  • 离散点最小(凸)包围边界查找
  • 让你成为前端,后端或全栈开发程序员的进阶指南,一门学到老的技术
  • 小程序 setData 学问多
  • 鱼骨图 - 如何绘制?
  • 400多位云计算专家和开发者,加入了同一个组织 ...
  • !!Dom4j 学习笔记
  • # Pytorch 中可以直接调用的Loss Functions总结:
  • # 数据结构
  • $.each()与$(selector).each()
  • ( )的作用是将计算机中的信息传送给用户,计算机应用基础 吉大15春学期《计算机应用基础》在线作业二及答案...
  • (C语言)fread与fwrite详解
  • (vue)el-checkbox 实现展示区分 label 和 value(展示值与选中获取值需不同)
  • (算法)Game
  • (小白学Java)Java简介和基本配置
  • (译)计算距离、方位和更多经纬度之间的点
  • (转)ABI是什么
  • (转载)hibernate缓存
  • (转载)PyTorch代码规范最佳实践和样式指南
  • .net core 控制台应用程序读取配置文件app.config
  • .NET 常见的偏门问题
  • .net 获取url的方法
  • .net6+aspose.words导出word并转pdf
  • .NET上SQLite的连接
  • .NET中使用Protobuffer 实现序列化和反序列化
  • @Bean, @Component, @Configuration简析
  • @JsonFormat与@DateTimeFormat注解的使用
  • @RequestBody详解:用于获取请求体中的Json格式参数
  • @RequestMapping处理请求异常
  • @RestController注解的使用