并发编程(一)-Thread 源码分析
创始人
2025-05-30 13:35:25
0

一、什么是线程


线程(英语:thread)是操作系统能够进行运算调度的最小单位。它被包含在进程之中,是进程中的实际运作单位。一条线程指的是进程中一个单一顺序的控制流,一个进程中可以并发多个线程,每条线程并行执行不同的任务。在Unix System V及SunOS中也被称为轻量进程(lightweight processes),但轻量进程更多指内核线程(kernel thread),而把用户线程(user thread)称为线程。

二、线程创建方式


2.1 继承Thread

public class MyThread extends Thread{@Overridepublic void run(){}
}

2.2 实现Runnable接口

public class MyThread implements Runnable(){@Overridepublic void run(){}
}

2.3 实现Callable接口

public class MyThread implements Callable {@Overridepublic Integer call() throws Exception {return 0;}
}

从Diagrams 图中就可得知,不管是extends Thread 还是 implements Callable接口,底层归根到底还是,实现了Runnable接口。

三、源码分析


3.1 线程中属性含义


// 线程名称
private volatile String name;
// 线程优先级
private int            priority;
// 内置的Thread类
private Thread         threadQ;
// JVM中的java thread 指针
private long           eetop;// 是否是单步执行此线程
private boolean     single_step;// 是否是守护线程
private boolean     daemon = false;// jvm 状态
private boolean     stillborn = false;// 构造函数中会传入的线程对象
private Runnable target;// 线程组
private ThreadGroup group;// 类加载器
private ClassLoader contextClassLoader;// 继承的访问下文
private AccessControlContext inheritedAccessControlContext;// 静态变量,使用内部静态类的单例模式,全局存在,用来生成线程名
private static int threadInitNumber;
private static synchronized int nextThreadNum() {return threadInitNumber++;
}// ThreadLocal 能为线程设置线程私有变量,设置线程上下文
ThreadLocal.ThreadLocalMap threadLocals = null;// inheritableThreadLocals 解决子线程能够获取父线程变量
ThreadLocal.ThreadLocalMap inheritableThreadLocals = null;// 给线程分配的栈大小,线程默认栈大小为0
private long stackSize;// 本地线程终止之后的专用状态
private long nativeParkEventPointer;// 线程id
private long tid;// 用于生成线程id,新创建线程后自增
private static long threadSeqNumber;// 线程状态,初始线程状态:0
private volatile int threadStatus = 0;// 生成线程id方法
private static synchronized long nextThreadID() {return ++threadSeqNumber;
}// LockSupport 中断
volatile Object parkBlocker;// interrupt 相关
private volatile Interruptible blocker;
private final Object blockerLock = new Object();
void blockedOn(Interruptible b) {synchronized (blockerLock) {blocker = b;}
}// 线程最低优先级
public final static int MIN_PRIORITY = 1;// 线程默认优先级
public final static int NORM_PRIORITY = 5;// 线程最高优先级
public final static int MAX_PRIORITY = 10;

3.2 线程状态

public enum State {// 新建状态:Thread t = new Thread();NEW,// 运行,调用了t.start(),此时就绪(等待CPU进行调度)和运行都属于运行状态。RUNNABLE,// 阻塞,因为争用 synchronized 的 monitor 对象而发生阻塞的线程处于 blocked 状态。BLOCKED,// 等待,需要其他的线程进行唤醒。一般是调用Object.wait,Thread.join,LockSupport.parkWAITING。可以调用Object.notify,Object.notifyAll,LockSupport.unpark进行唤醒// 超时等待,可以在指定时间内自动唤醒。调用Thread.sleep(long),Object.wait(long),Thread.join(long),// LockSupport.parkNanos(long),LockSupport.parkUntil(long)等进入TIMED_WAITING状态。// 唤醒方式:时间到了,Object.notify,Object.notifyAll,LockSupport.unpark等方法TIMED_WAITING,// 终止线程的线程状态。线程已完成执行。TERMINATED;
}

3.3构造方法

3.3.1 jdk1.8中Thread

3.4 init方法


private void init(ThreadGroup g, Runnable target, String name,long stackSize, AccessControlContext acc,boolean inheritThreadLocals) {// 线程名称校验if (name == null) {throw new NullPointerException("name cannot be null");}this.name = name;// 父线程Thread parent = currentThread();// 获取安全管理器SecurityManager security = System.getSecurityManager();// 线程组为nullif (g == null) {// 安全管理器不为null,从安全管理器中获取线程组if (security != null) {g = security.getThreadGroup();}// 线程组为null,则从父线程中获取线程组if (g == null) {g = parent.getThreadGroup();}}// 检查当前线程拥有线程组权限g.checkAccess();if (security != null) {if (isCCLOverridden(getClass())) {security.checkPermission(SUBCLASS_IMPLEMENTATION_PERMISSION);}}// 添加未启动线程数g.addUnstarted();this.group = g;// 设置守护线程标识this.daemon = parent.isDaemon();// 获取线程优先级this.priority = parent.getPriority();// 设置类加载器if (security == null || isCCLOverridden(parent.getClass()))this.contextClassLoader = parent.getContextClassLoader();elsethis.contextClassLoader = parent.contextClassLoader;   this.inheritedAccessControlContext =acc != null ? acc : AccessController.getContext();this.target = target;// 设置线程优先级setPriority(priority);if (inheritThreadLocals && parent.inheritableThreadLocals != null)// 子线程集成父线程的ThreadLocalthis.inheritableThreadLocals =ThreadLocal.createInheritedMap(parent.inheritableThreadLocals);// 设置线程栈大小this.stackSize = stackSize;// 设置线程idtid = nextThreadID();
}

3.5 start方法

public synchronized void start() {// 检查线程状态是否是初始化状态( 0 == NEW)if (threadStatus != 0)throw new IllegalThreadStateException();// 线程组中添加该线程group.add(this);boolean started = false;try {// 调用native方法start0();// 成功启动标识started = true;} finally {try {// 没有启动成功if (!started) {// 从线程组中移除启动失败的线程group.threadStartFailed(this);}} catch (Throwable ignore) {}}
}private native void start0()

3.6 run方法


public void run() {if (target != null) {// 调用构造函数Runnable的run方法target.run();}
}

3.7 interrupt方法

interrupt并不是强制中断停止线程,仅仅更改线程状态。被设置中断标志的线程将继续正常运行,不受影响。


public void interrupt() {if (this != Thread.currentThread())checkAccess();synchronized (blockerLock) {Interruptible b = blocker;if (b != null) {// 设置中断状态interrupt0();         // 调用阻断程序中的中断方法b.interrupt(this);return;}}interrupt0();
}

3.8 enumerate方法

// 将当前线程组及其子线程组的线程复制到指定数组中
public static int enumerate(Thread tarray[]) {return currentThread().getThreadGroup().enumerate(tarray);
}

3.9 join(long millis)方法


public final synchronized void join(long millis)throws InterruptedException {long base = System.currentTimeMillis();long now = 0;// 等待毫秒数为负数,则抛出异常  if (millis < 0) {throw new IllegalArgumentException("timeout value is negative");}// 等待毫秒数等于0,会一直等待,直到线程死亡,原理就是自旋等待if (millis == 0) {while (isAlive()) {wait(0);}} else {// 等待毫秒数大于0,超出等待时间,则退出。超时自旋等待while (isAlive()) {long delay = millis - now;// 如果等待时间小于等于0,退出等待if (delay <= 0) {break;}// 调用wait方法进行线程等待wait(delay);now = System.currentTimeMillis() - base;}}
}

3.10 sleep(long millis, int nanos)方法

线程休眠,线程休眠会将线程的cpu释放,但是线程占有的锁不会释放,当前线程进入Timed_waiting。

public static void sleep(long millis, int nanos)throws InterruptedException {// 睡眠毫秒实际小于0,抛出异常if (millis < 0) {throw new IllegalArgumentException("timeout value is negative");}// 睡眠纳秒时间 0~999999,if (nanos < 0 || nanos > 999999) {throw new IllegalArgumentException("nanosecond timeout value out of range");}// 1毫秒 = 1000000纳秒,纳秒数据超过一般0.5毫秒数据,就当做1毫秒,或者毫秒是0 而毫秒非零,也加1if (nanos >= 500000 || (nanos != 0 && millis == 0)) {millis++;}// 调用native方法,线程休眠sleep(millis);
}

3.11 yield方法

yield是 Thread 类中的native方法,作用:让出当前线程CPU的时间片,使当前线程从执行状态变为就绪状态,cpu将会从就绪队列中重新选择一个线程,也就是说,当前线程还是有可能会被再次选择的。

相关内容

热门资讯

安卓系统进不去无命令,安卓系统... 手机屏幕上突然黑屏了,安卓系统怎么就进不去了呢?别急,别慌,今天就来给你详细解析一下这个问题,让你轻...
适合安卓系统k歌软件,打造个人... 你有没有想过,在手机上也能尽情地唱出你的心声呢?现在,就让我带你走进一个神奇的世界,那就是适合安卓系...
安卓系统怎么充电的视频 手机电量告急,又到了充电的时刻啦!你是不是也和我一样,对安卓系统的充电方式充满了好奇?今天,就让我带...
ios系统与安卓系统的内存对比... 你有没有发现,手机里的世界越来越精彩了?各种应用层出不穷,游戏、社交、办公,样样都离不开手机。而支撑...
安卓7.1系统打开usb方法,... 你有没有想过,有时候你的安卓手机就像一个神秘的宝盒,里面藏着许多你意想不到的小秘密?今天,我就要给你...
安卓系统如何删不了游戏,揭秘解... 你是不是也遇到了这样的烦恼?手机里游戏堆积如山,想删删不掉,真是让人头疼啊!别急,今天就来跟你聊聊安...
电脑如何重置安卓手机系统,轻松... 你是不是也遇到了安卓手机卡顿、运行缓慢的问题?别急,今天就来教你怎么用电脑轻松重置安卓手机系统,让你...
origin怎么进入安卓系统,... 你有没有想过,你的安卓手机里隐藏着无数的秘密?今天,就让我带你一探究竟,揭开“origin怎么进入安...
索尼电视安卓系统关闭 最近家里那台索尼电视闹了个小风波,安卓系统竟然突然关闭了!这可真是让人摸不着头脑,毕竟这可是咱们家的...
怎么弄旧安卓系统,旧安卓系统升... 你手上的安卓手机是不是已经有点儿“古董”的味道了?别急,别急,今天就来教你怎么给它来个“青春焕发”的...
安卓系统刷成欧版,体验纯正欧洲... 你有没有想过,让你的安卓手机焕然一新,体验一把欧洲风格的魅力呢?没错,就是那种一打开手机,就能感受到...
怎么控制安卓系统定位,安全无忧 手机里的定位功能有时候真是让人又爱又恨呢!有时候,我们希望它精准地知道我们的位置,比如导航时;但有时...
安卓系统如何优化续航,解锁手机... 手机电量总是不够用,是不是你也和我一样,每天都要带着充电宝出门?别急,今天就来和你聊聊安卓系统如何优...
安卓仿苹果所有系统,从界面到体... 你有没有发现,现在手机市场上安卓仿苹果系统的手机越来越多了?没错,就是那种长得像苹果,用起来也像苹果...
安卓系统升级固件下载,轻松实现... 亲爱的手机控们,你是不是也和我一样,对安卓系统的每一次升级都充满了期待呢?每次系统更新,都仿佛是给我...
安卓系统可以放几个软件,无限拓... 你有没有想过,你的安卓手机里能装多少个软件呢?这可是个挺有意思的话题呢!想象从早晨起床到晚上睡觉,你...
优酷安卓9.0系统版本,畅享流... 你有没有发现,最近你的优酷APP是不是有点不一样了?没错,就是那个我们每天离不开的追剧神器——优酷,...
安卓手机系统体验排名,揭秘最佳... 你有没有发现,现在手机市场上安卓手机的品牌和型号简直多到让人眼花缭乱?每个品牌都试图在系统体验上大显...
安卓操作系统技巧在哪,安卓操作... 你有没有发现,安卓手机用久了,总感觉有点慢吞吞的?别急,今天就来给你支几招,让你的安卓手机焕发第二春...
安卓手机哪个系统最快,揭秘最快... 你有没有想过,为什么你的安卓手机有时候会慢吞吞的,像是老牛拉破车一样?别急,今天就来给你揭秘安卓手机...