经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » Java » 查看文章
Java并发(九)----线程join、interrupt
来源:cnblogs  作者:|旧市拾荒|  时间:2023/6/2 10:51:50  对本文有异议

1、join 方法详解

1.1 为什么需要 join?

下面的代码执行,打印 r 是什么?

  1. static int r = 0;
  2. public static void main(String[] args) throws InterruptedException {
  3.    test1();
  4. }
  5. private static void test1() throws InterruptedException {
  6.    log.debug("开始");
  7.    Thread t1 = new Thread(() -> {
  8.        log.debug("开始");
  9.        sleep(1);
  10.        log.debug("结束");
  11.        r = 10;
  12.   });
  13.    t1.start();
  14.    log.debug("结果为:{}", r);
  15.    log.debug("结束");
  16. }

分析

  • 因为主线程和线程 t1 是并行执行的,t1 线程需要 1 秒之后才能算出 r=10

  • 而主线程一开始就要打印 r 的结果,所以只能打印出 r=0

解决方法

  • 用 主线程sleep 行不行?为什么? 这种方式不推荐,因为不清楚t1线程执行具体的时间

  • 用 join,加在 t1.start() 之后即可,主线程执行到t1.join()时会等待t1线程结束

1.2 等待单个结果

以调用方角度来讲,如果

  • 需要等待结果返回,才能继续运行就是同步

  • 不需要等待结果返回,就能继续运行就是异步

1.2 等待多个结果

问,下面代码 cost 大约多少秒?

  1. static int r1 = 0;
  2. static int r2 = 0;
  3. public static void main(String[] args) throws InterruptedException {
  4.    test2();
  5. }
  6. private static void test2() throws InterruptedException {
  7.    Thread t1 = new Thread(() -> {
  8.        sleep(1);
  9.        r1 = 10;
  10.   });
  11.    Thread t2 = new Thread(() -> {
  12.        sleep(2);
  13.        r2 = 20;
  14.   });
  15.    long start = System.currentTimeMillis();
  16.    t1.start();
  17.    t2.start();
  18.    t1.join();
  19.    t2.join();
  20.    long end = System.currentTimeMillis();
  21.    log.debug("r1: {} r2: {} cost: {}", r1, r2, end - start);
  22. }

分析如下

  • 第一个 join:等待 t1 时, t2 并没有停止, 而在运行

  • 第二个 join:1s 后, 执行到此, t2 也运行了 1s, 因此也只需再等待 1s

如果颠倒两个 join 呢?

最终都是输出

  1. 20:45:43.239 [main] c.TestJoin - r1: 10 r2: 20 cost: 2005

1.3 有时效的 join

等够时间

  1. static int r1 = 0;
  2. static int r2 = 0;
  3. public static void main(String[] args) throws InterruptedException {
  4.    test3();
  5. }
  6. public static void test3() throws InterruptedException {
  7.    Thread t1 = new Thread(() -> {
  8.        sleep(1);
  9.        r1 = 10;
  10.   });
  11. ?
  12.    long start = System.currentTimeMillis();
  13.    t1.start();
  14. ?
  15.    // 线程执行结束会导致 join 结束
  16.    t1.join(1500);
  17.    long end = System.currentTimeMillis();
  18.    log.debug("r1: {} r2: {} cost: {}", r1, r2, end - start);
  19. }

输出

  1. 20:48:01.320 [main] c.TestJoin - r1: 10 r2: 0 cost: 1010

没等够时间

  1. static int r1 = 0;
  2. static int r2 = 0;
  3. public static void main(String[] args) throws InterruptedException {
  4.    test3();
  5. }
  6. public static void test3() throws InterruptedException {
  7.    Thread t1 = new Thread(() -> {
  8.        sleep(2);
  9.        r1 = 10;
  10.   });
  11. ?
  12.    long start = System.currentTimeMillis();
  13.    t1.start();
  14. ?
  15.    // 线程执行结束会导致 join 结束
  16.    t1.join(1500);
  17.    long end = System.currentTimeMillis();
  18.    log.debug("r1: {} r2: {} cost: {}", r1, r2, end - start);
  19. }

输出

  1. 20:52:15.623 [main] c.TestJoin - r1: 0 r2: 0 cost: 1502

2、interrupt 方法详解

其主要作用是打断 sleep,wait,join 的线程

这几个方法都会让线程进入阻塞状态

打断 sleep 的线程, 会清空打断状态,以 sleep 为例

  1. private static void test1() throws InterruptedException {
  2.    Thread t1 = new Thread(()->{
  3.        sleep(1);
  4.   }, "t1");
  5.    t1.start();
  6. ?
  7.    sleep(0.5);
  8.    t1.interrupt();
  9.    log.debug(" 打断状态: {}", t1.isInterrupted());
  10. }

输出

  1. java.lang.InterruptedException: sleep interrupted
  2. at java.lang.Thread.sleep(Native Method)
  3. at java.lang.Thread.sleep(Thread.java:340)
  4. at java.util.concurrent.TimeUnit.sleep(TimeUnit.java:386)
  5. at cn.itcast.n2.util.Sleeper.sleep(Sleeper.java:8)
  6. at cn.itcast.n4.TestInterrupt.lambda$test1$3(TestInterrupt.java:59)
  7. at java.lang.Thread.run(Thread.java:745)
  8. 21:18:10.374 [main] c.TestInterrupt - 打断状态: false

正常运行状态的如果打断,那么打断标记为true,如果是阻塞状态被打断,那么其打断状态为false。

2.1 打断正常运行的线程

打断正常运行的线程, 不会清空打断状态

  1. private static void test2() throws InterruptedException {
  2.    Thread t2 = new Thread(()->{
  3.        while(true) {
  4.            Thread current = Thread.currentThread();
  5.            boolean interrupted = current.isInterrupted();
  6.            if(interrupted) {
  7.                log.debug(" 打断状态: {}", interrupted);
  8.                break;
  9.           }
  10.       }
  11.   }, "t2");
  12.    t2.start();
  13. ?
  14.    sleep(0.5);
  15.    t2.interrupt();
  16. }

输出

  1. 20:57:37.964 [t2] c.TestInterrupt - 打断状态: true

注意:这个打断标记只是一个标记信号,并不会结束线程的执行,一般是根据这个标记信号来决定是否结束当前线程。

2.2 采用两阶段终止线程,避免stop停止

在一个线程 T1 中如何“优雅”终止线程 T2?这里的【优雅】指的是给 T2 一个料理后事的机会。

错误思路

  • 使用线程对象的 stop() 方法停止线程

    • stop 方法会真正杀死线程,如果这时线程锁住了共享资源,那么当它被杀死后就再也没有机会释放锁, 其它线程将永远无法获取锁

  • 使用 System.exit(int) 方法停止线程

    • 目的仅是停止一个线程,但这种做法会让整个程序都停止

两阶段终止模式

2.2.1 利用 isInterrupted

interrupt 可以打断正在执行的线程,无论这个线程是在 sleep,wait,还是正常运行

  1. class TPTInterrupt {
  2.    private Thread thread;
  3.    public void start(){
  4.        thread = new Thread(() -> {
  5.            while(true) {
  6.                Thread current = Thread.currentThread();
  7.                if(current.isInterrupted()) {
  8.                    log.debug("料理后事");
  9.                    break;
  10.               }
  11.                try {
  12.                    Thread.sleep(1000);
  13.                    log.debug("将结果保存");
  14.               } catch (InterruptedException e) {
  15.                    current.interrupt();  // 设置为true
  16.               }
  17.                // 执行监控操作
  18.           }
  19.       },"监控线程");
  20.        thread.start();
  21.   }
  22.    public void stop() {
  23.        thread.interrupt();
  24.   }
  25. }

调用

  1. TPTInterrupt t = new TPTInterrupt();
  2. t.start();
  3. Thread.sleep(3500);
  4. log.debug("stop");
  5. t.stop();

结果

  1. 11:49:42.915 c.TwoPhaseTermination [监控线程] - 将结果保存
  2. 11:49:43.919 c.TwoPhaseTermination [监控线程] - 将结果保存
  3. 11:49:44.919 c.TwoPhaseTermination [监控线程] - 将结果保存
  4. 11:49:45.413 c.TestTwoPhaseTermination [main] - stop
  5. 11:49:45.413 c.TwoPhaseTermination [监控线程] - 料理后事
2.2.2 利用停止标记
  1. // 停止标记用 volatile 是为了保证该变量在多个线程之间的可见性
  2. // 我们的例子中,即主线程把它修改为 true 对 t1 线程可见
  3. class TPTVolatile {
  4.    private Thread thread;
  5.    private volatile boolean stop = false;
  6.    public void start(){
  7.        thread = new Thread(() -> {
  8.            while(true) {
  9.                Thread current = Thread.currentThread();
  10.                if(stop) {
  11.                    log.debug("料理后事");
  12.                    break;
  13.               }
  14.                try {
  15.                    Thread.sleep(1000);
  16.                    log.debug("将结果保存");
  17.               } catch (InterruptedException e) {
  18.               }
  19.                // 执行监控操作
  20.           }
  21.       },"监控线程");
  22.        thread.start();
  23.   }
  24.    public void stop() {
  25.        stop = true;
  26.        thread.interrupt();
  27.   }
  28. }

调用

  1. TPTVolatile t = new TPTVolatile();
  2. t.start();
  3. Thread.sleep(3500);
  4. log.debug("stop");
  5. t.stop();

结果

  1. 11:54:52.003 c.TPTVolatile [监控线程] - 将结果保存
  2. 11:54:53.006 c.TPTVolatile [监控线程] - 将结果保存
  3. 11:54:54.007 c.TPTVolatile [监控线程] - 将结果保存
  4. 11:54:54.502 c.TestTwoPhaseTermination [main] - stop
  5. 11:54:54.502 c.TPTVolatile [监控线程] - 料理后事

2.3 打断处于park状态线程

park, 进入WAITING状态,对比wait不需要获得锁就可以让线程WAITING,通过unpark唤醒

打断 处于park状态 线程, 不会清空打断状态

  1. private static void test3() throws InterruptedException {
  2.    Thread t1 = new Thread(() -> {
  3.        log.debug("park...");
  4.        LockSupport.park();
  5.        log.debug("unpark...");
  6.        log.debug("打断状态:{}", Thread.currentThread().isInterrupted());
  7.   }, "t1");
  8.    t1.start();
  9. ?
  10. ?
  11.    sleep(1);
  12.    t1.interrupt();
  13. }

输出

  1. 21:11:52.795 [t1] c.TestInterrupt - park...
  2. 21:11:53.295 [t1] c.TestInterrupt - unpark...
  3. 21:11:53.295 [t1] c.TestInterrupt - 打断状态:true

如果打断标记已经是 true, 则 park 会失效

  1. private static void test4() {
  2.    Thread t1 = new Thread(() -> {
  3.        for (int i = 0; i < 5; i++) {
  4.            log.debug("park...");
  5.            LockSupport.park();
  6.            log.debug("打断状态:{}", Thread.currentThread().isInterrupted());
  7.       }
  8.   });
  9.    t1.start();
  10. ?
  11. ?
  12.    sleep(1);
  13.    t1.interrupt();
  14. }

输出

  1. 21:13:48.783 [Thread-0] c.TestInterrupt - park...
  2. 21:13:49.809 [Thread-0] c.TestInterrupt - 打断状态:true
  3. 21:13:49.812 [Thread-0] c.TestInterrupt - park...
  4. 21:13:49.813 [Thread-0] c.TestInterrupt - 打断状态:true
  5. 21:13:49.813 [Thread-0] c.TestInterrupt - park...
  6. 21:13:49.813 [Thread-0] c.TestInterrupt - 打断状态:true
  7. 21:13:49.813 [Thread-0] c.TestInterrupt - park...
  8. 21:13:49.813 [Thread-0] c.TestInterrupt - 打断状态:true
  9. 21:13:49.813 [Thread-0] c.TestInterrupt - park...
  10. 21:13:49.813 [Thread-0] c.TestInterrupt - 打断状态:true

提示

可以使用 Thread.interrupted() 清除打断状态

3、不推荐的方法

还有一些不推荐使用的方法,这些方法已过时,容易破坏同步代码块,造成线程死锁

方法名 static 功能说明
stop()   停止线程运行
suspend()   挂起(暂停)线程运行
resume()   恢复线程运行

 

 

原文链接:https://www.cnblogs.com/xiaoyh/p/17092356.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号