Java 线程

142 阅读3分钟

这是我参与11月更文挑战的第10天,活动详情查看:2021最后一次更文挑战

jvm线程

jvm实现线程需要考虑如何使用操作系统的线程。在jdk发行之初,采用的是用户线程,即多个jvm线程对应到一个操作系统线程,也称作绿色线程。但其存在的问题是,一些针对操作系统的线程监控工具并不适用,虽然jvm有自己的jstack工具,但无法对cpu占用进行查看,所以在后续版本中,jvm去掉了绿色线程,用户线程与操作系统的为1比1的关系。

其实我倒是感觉绿色线程有点协程的意思。

jvm线程状态

在jdk代码中,定义了线程的几种状态:


public enum State {
        /**
         * Thread state for a thread which has not yet started.
         * new Thread()后,线程所处状态
         */
        NEW,

        /**
         * Thread state for a runnable thread.  A thread in the runnable
         * state is executing in the Java virtual machine but it may
         * be waiting for other resources from the operating system
         * such as processor.
         * 当执行thrad.start()方法后,线程所处状态,有些博客也将该状态分为runnable(线程等待CPU资源)与running(线程获取到CPU执行权)状态
         */
        RUNNABLE,

        /**
         * Thread state for a thread blocked waiting for a monitor lock.
         * A thread in the blocked state is waiting for a monitor lock
         * to enter a synchronized block/method or
         * reenter a synchronized block/method after calling
         * {@link Object#wait() Object.wait}.
         上面这段文字描述了几个让线程进入BLOCKED状态的场景
         1.线程在进入synchronized代码时争夺锁重量级锁失败导致线程挂起进入BLOCKED
         2.当线程从Object#wait()恢复后重新争抢重量级锁失败线程进入BLOCKED状态
         BLOCKED的定义为线程竞争锁失败
         */
        BLOCKED,

        /**
         * Thread state for a waiting thread.
         * A thread is in the waiting state due to calling one of the
         * following methods:
         * <ul>
         *   <li>{@link Object#wait() Object.wait} with no timeout</li>
         *   <li>{@link #join() Thread.join} with no timeout</li>
         *   <li>{@link LockSupport#park() LockSupport.park}</li>
         * </ul>
         *
         * <p>A thread in the waiting state is waiting for another thread to
         * perform a particular action.
         *
         * For example, a thread that has called <tt>Object.wait()</tt>
         * on an object is waiting for another thread to call
         * <tt>Object.notify()</tt> or <tt>Object.notifyAll()</tt> on
         * that object. A thread that has called <tt>Thread.join()</tt>
         * is waiting for a specified thread to terminate.
         waiting状态用于表示线程等待另一个线程某个特定动作的执行
         上面说道三种情况进入WATING状态:
         1. Object#wait()
         2. Thread.join()
         3. LockSupport#park()
         */
        WAITING,

        /**
         * Thread state for a waiting thread with a specified waiting time.
         * A thread is in the timed waiting state due to calling one of
         * the following methods with a specified positive waiting time:
         * <ul>
         *   <li>{@link #sleep Thread.sleep}</li>
         *   <li>{@link Object#wait(long) Object.wait} with timeout</li>
         *   <li>{@link #join(long) Thread.join} with timeout</li>
         *   <li>{@link LockSupport#parkNanos LockSupport.parkNanos}</li>
         *   <li>{@link LockSupport#parkUntil LockSupport.parkUntil}</li>
         * </ul>
         
         TIMED_WAITING表示线程等待其它线程的某个特定动作,但是只在限定时间等待
         以下方法使线程进入TIMED_WAITTING
         1. Thread.sleep()
         2. Object#wait(long)
         3. LockSupport#parkNanos(long)
         4. LockSupport#parkUntil(long)
         5. Thread.join(long)
         */
        TIMED_WAITING,

        /**
         * Thread state for a terminated thread.
         * The thread has completed execution.
         
         TERMINATED用于表示一个线程已经执行完成
         */
        TERMINATED;
    }

需要注意的是BLOCKED状态,《并发编程的艺术》中没有提到Object#wait()后线程重新竞争重量级锁,也会进入BLOCKED状态,这也是容易忽视的一个点,从 BLOCKED状态表示线程竞争锁失败进入阻塞,会容易记住些。

感觉没啥可写了,下面说一下自己实际开发遇到的一个问题,使用线程异步执行时如何让traceId透传过下去。以下是我在项目中的使用方式:

public class TraceContext {
    public static ThreadLocal<String> traceContext = new ThreadLocal<>();
}
public class NewObjectTest {
    
    public static void main(String[] args) throws InterruptedException {
        // 设置当前线程的trace
        TraceContext.traceContext.set("main-traceId");
        Runnable r = () -> {
            // 设置子线程trace
            TraceContext.traceContext.set(TraceContext.traceContext.get());
            task();
        };
        Thread t = new Thread();
    }

    private static void task() {
        System.out.println(TraceContext.traceContext.get());
    }
}

好像阿里也有开源ThreadLocal透传的工具,大家用的什么方法呢,欢迎分享。