标签:虚拟 lan 接收 初始 art getname 重写 illegal 思维
在前面我们已经了解到如何创建线程,要么就重写Thread类的run方法,要么就实现Runnable接口的run方法,然后调用Thread类的start方法去启动它,可能我们发现了,最终执行的其实还是run方法里面的代码,那么为什么一定要通过start方法来调用呢?我直接调用run方法不就好了吗,正常思维情况下,确实是这样的,但是到底对不对,我们来做个试验;
代码如下:
public static void main(String[] args) {
Runnable runnable = () -> {
System.out.println(Thread.currentThread().getName());
};
runnable.run();
new Thread(runnable).start();
}
我们先创建了一个Runnable对象并实现了它的run方法,然后直接通过对象.run()来调用,
后面的是我们创建了一个Thread类,并将runnable对象传了进去然后调用start方法。
如果按照我们前面设想的一样,那么执行之后控制台应该打印出两个新的线程的名字,但是实际的打印结果是:
main
Thread-0
这与我们预期的不同,直接调用run方法,打印的是主线程的名字,而不是我们设想的那样,是一个新的线程的名字,这是为什么?
这是因为在Java中,代码的执行都是通过主线程来执行的,而run方法其实就是一个普通的Java方法,你直接调用run方法,那么程序就直接执行了,并不会去开启一个新的线程,那为什么start方法就可以开启一个新的线程呢?我们来看看start方法的含义
我们知道,代码都是通过主线程来一行行的执行的,当代码执行到Thread.start方法这一行时,主线程就发现,咦,这个家伙想要开启一个新的线程,那好的吧,我去通知一下虚拟机(JVM),这里有个家伙想要开启一个新线程了,你有没有空,有空就帮它开启一下吧,然后虚拟机接收到这个请求后,等到空闲时,就满足了它的这个要求。
那这个时候我们就知道了,start这个方法是用来通知虚拟机,我想要开启一个新的线程,这个时候虚拟机在条件允许的情况下,就会满足它的要求。同时我们还需要注意一个点,真正发出通知的是主线程,是它来帮我们发出这个请求的。所以,正确启动线程的方式是调用Thread类的start方法。
我就提前告诉你好了,会抛出一个IllegalThreadStateException的异常,至于是为什么,就需要我们去看看Thread类的源码了,我们进入到Thread类中的start方法后,会发现,在方法的一开始,就会做一个判断:
if (threadStatus != 0)
throw new IllegalThreadStateException();
如果threadStatus != 0 的话就抛出这个异常了,threadStatus又是什么,它代表了一个线程的状态,线程有6个状态,至于是哪6个状态,后续会讲的。在Thread类初始化的时候,threadStatus就已经被赋值为0了,当我们调用了start方法之后,这个状态值就会被修改,所以,当我们重复调用start方法的时候,就会抛出异常了
标签:虚拟 lan 接收 初始 art getname 重写 illegal 思维
原文地址:https://www.cnblogs.com/islcz/p/xian-cheng-ru-he-zheng-que-de-qi-dong.html