zoukankan      html  css  js  c++  java
  • 并发和多线程(三)--线程的生命周期/状态

      线程的生命周期或者说状态其实不复杂,但是很多人的理解可能有错误,一个典型的误区,线程运行时的状态是Runnable,而不是Running,因为线程没有Running状态。

    线程的状态

    1、New:已创建,没启动。还没有执行start()

    2、Runnable:调用start()之后就处于Runnable,无论是否已经运行,都是Runnable状态,对应操作系统的Ready和Running状态。

    3、Blocked:进入Synchronized修饰的方法或者代码块,但是无法获取锁,就处于Blocked。

    4、Waiting:线程进入等待的阻塞状态,例如调用Object.wait()。

    5、Timed-Waiting:线程进入计时等待的阻塞状态,例如调用Thread.sleep(time)。

    6、Terminated:线程执行代码结束,或者出现未捕捉的异常。

    PS:线程没有Running状态,请参考官方文档:https://docs.oracle.com/javase/7/docs/api/java/lang/Thread.State.html

    线程的状态转换

      从上图,可以明确看到状态如何进行转换,状态之间的变化是否可逆。。。

      其实,从Waiting到Blocked状态可以可以的,当Waiting状态被唤醒之后,如果暂时没有获取到monitor锁,这时候就会进入Blocked状态,当获取锁之后,重新变成Runnable。Waiting和Timed-Waiting状态下在出现未捕获异常时,就会直接进入Terminated。

      PS:Waiting和Timed-Waiting状态下通过interrupt()响应中断,会进入Runnable状态。

    验证线程状态

      为了上面所讲线程状态的可信度,我们通过代码进行验证

    1、New、Runnable、Terminated

    public static void main(String[] args) throws InterruptedException{
        Thread thread = new Thread(() -> {
            for (int i = 0; i <= 4; i++) {
                log.info("{}", i);
                if (i == 2) {
                    log.info("子线程运行过程中状态:{}", Thread.currentThread().getState());
                }
            }
        });
    
        log.info("子线程没有执行start时,状态:{}", thread.getState());
        thread.start();
        log.info("子线程执行start后,状态:{}", thread.getState());
        Thread.sleep(10);
        log.info("子线程执行结束,状态:{}", thread.getState());
    }
    结果:
    17:12:11.244 [main] INFO com.diamondshine.Thread.ThreadClass - 子线程没有执行start时,状态:NEW
    17:12:11.251 [main] INFO com.diamondshine.Thread.ThreadClass - 子线程执行start后,状态:RUNNABLE
    17:12:11.251 [Thread-0] INFO com.diamondshine.Thread.ThreadClass - 0
    17:12:11.251 [Thread-0] INFO com.diamondshine.Thread.ThreadClass - 1
    17:12:11.251 [Thread-0] INFO com.diamondshine.Thread.ThreadClass - 2
    17:12:11.252 [Thread-0] INFO com.diamondshine.Thread.ThreadClass - 子线程运行过程中状态:RUNNABLE
    17:12:11.252 [Thread-0] INFO com.diamondshine.Thread.ThreadClass - 3
    17:12:11.252 [Thread-0] INFO com.diamondshine.Thread.ThreadClass - 4
    17:12:11.262 [main] INFO com.diamondshine.Thread.ThreadClass - 子线程执行结束,状态:TERMINATED

      成功验证了在子线程执行过程中的状态是Runnable,而不是Running。

    2、Blocked、Waiting、Time-Waiting

    @Slf4j
    public class ThreadClass implements Runnable{
    
        public static void main(String[] args) throws InterruptedException{
    
            ThreadClass threadClass = new ThreadClass();
            Thread thread = new Thread(threadClass);
            thread.start();
            Thread thread1 = new Thread(threadClass);
            thread1.start();
    
            //sleep 5ms为了让子线程执行到synchronize()的Thread.sleep(1000)
            try {
                Thread.sleep(5);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
    
            log.info("第一个线程的状态:{}", thread.getState());
            //此时第一个线程获取monitor锁,然后休眠1000ms,所以第二个线程无法获得锁
            log.info("第二个线程的状态:{}", thread1.getState());
    
            //休眠1100ms为了让代码执行到wait();
            Thread.sleep(1100);
            log.info("第一个线程的状态:{}", thread.getState());
        }
    
        @Override
        public void run() {
            synchronize();
        }
    
        private synchronized void synchronize() {
            try {
                Thread.sleep(1000);
                wait();
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
                e.printStackTrace();
            }
        }
    }
    结果:
    17:46:19.601 [main] INFO com.diamondshine.Thread.ThreadClass - 第一个线程的状态:TIMED_WAITING
    17:46:19.608 [main] INFO com.diamondshine.Thread.ThreadClass - 第二个线程的状态:BLOCKED
    17:46:20.708 [main] INFO com.diamondshine.Thread.ThreadClass - 第一个线程的状态:WAITING

       验证结果符合预期,当调用sleep(),线程处于Timed_Waiting,因为Synchronized进入阻塞,处于Blocked,调用wait(),处于waiting状态。

  • 相关阅读:
    scala言语基础学习七
    scala言语基础学习六
    scala言语基础学习五
    scala言语基础学习四
    scala言语基础学习三(面向对象编程)
    scala言语基础学习三
    scala言语基础学习二
    scala言语基础学习
    并发编程实战的阅读(锁的重入)
    数据库必会必知 之 SQL四种语言:DDL DML DCL TCL(转)
  • 原文地址:https://www.cnblogs.com/huigelaile/p/11721305.html
Copyright © 2011-2022 走看看