1. 线程的创建
我们在创建一个线程的时候都是这么做的
Thread thread = new Thread(() -> {
IntStream.range(1, 1000).forEach(item -> System.out.println(Thread.currentThread().getName() + "---->" + item));
});
直接使用的带有Runnable接口作为参数的构造器。直接进入这个构造器,进行查看,会看到Thread对象的任何构造方法都会调用init方法进行初始化。
private void init(ThreadGroup g, Runnable target, String name,
long stackSize) {
init(g, target, name, stackSize, null);
}
private void init(ThreadGroup g, Runnable target, String name,
long stackSize, AccessControlContext acc) {
if (name == null) {
throw new NullPointerException("name cannot be null");
}
this.name = name.toCharArray();
Thread parent = currentThread();
SecurityManager security = System.getSecurityManager();
......
}
在初始化的过程中,会对线程的各种属性进行设置。例如线程的名称,线程所属的线程组,是否是守护线程,线程的优先级,上下文加载器等。这里讲解几个需要注意的。
属性名 | 说明 |
---|---|
group | 线程组,允许线程访问有关其自己的线程组的信息,但不允许访问有关其线程组的父线程组或任何其他线程组的信息。 |
daemon | 是否作为守护线程,在主线程结束的时候,守护线程退出。默认的是父线程的属性 |
priority | 线程的优先级.默认的是父线程的优先级 |
contextClassLoader | 这个线程的类加载器。这个可以用来破坏双亲委派模型。 |
stackSize | 栈调用深度,默认为0。 |
2.线程的几种状态
线程的状态在Thread类中由枚举类State中表示
public enum State {
//尚未启动的线程的线程状态
NEW,
//可运行线程的线程状态
RUNNABLE,
//线程的线程状态被阻塞等待监视器锁定
BLOCKED,
//等待线程的线程状态
WAITING,
//具有指定等待时间的等待线程的线程状态。
TIMED_WAITING,
//终止线程的线程状态。
TERMINATED;
}
关于线程的状态之间的转换,网上很多相应的文章,这里在网上找一个
这个图的文章地址https://www.cnblogs.com/happy-coder/p/6587092.html,这里对这些方法进行说明:
方法 | 说明 |
---|---|
start | 使此线程调用该线程的run方法开始执行。调用start是创建一个线程,并调用run方法。如果只调用run,并不会创建线程,而是直接在当前线程中进行执行 |
yield | 当前线程让出cpu,进入RUNNABLE状态,但是不会释放锁。 |
sleep | 让当前线程进入TIMED_WAITING状态,当前线程不会退出cpu的时间片占用 |
join | 被当前线程join的线程进入TIMED_WAITING或者WAITING状态,被join的线程会退出cpu的占用 |
wait | 调用该方法的线程进入TIMED_WAITING或者WAITING状态,会释放锁 |
notify/notifyAll | 使调用了wait方法进入TIMED_WAITING或者WAITING状态的线程进入RUNNABLE状态 |
run | 运行这个线程的run方法 |
方法之间的比较
sleep | wait |
---|---|
不会释放锁 | 释放锁 |
yield | wait |
---|---|
不会释放锁 | 释放锁 |
进入RUNNABLE状态 | TIMED_WAITING或者WAITING状态 |
3.其余方法
方法 | 说明 |
---|---|
activeCount | 获取当前线程所在线程组及其子线程组中的活跃线程数量 |
holdsLock | 当前线程是否持有指定的锁对象 |
isAlive | 当前线程是否存活 |
getState | 获取当前线程状态 |
4.关于线程的start跟run方法的说明
先说一下这两个方法的区别
start | run |
---|---|
会调用底层jvm启动一个新的线程,并回调定义的run方法 | 不会启动一个新线程,只是回直接执行run方法 |
这里对于jvm创建线程进行一些说明,首先这里需要提到一个线程标准。POSIX线程简称Pthreads,标准定义了创建和操纵线程的一整套API。在Linux跟windows系统中都有对这个模型的实现,而在jvm的底层实现中也是使用这一套API进行的。