线程如何正确的启动

Posted islcz

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了线程如何正确的启动相关的知识,希望对你有一定的参考价值。

在前面我们已经了解到如何创建线程,要么就重写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方法的含义

start方法的含义

我们知道,代码都是通过主线程来一行行的执行的,当代码执行到Thread.start方法这一行时,主线程就发现,咦,这个家伙想要开启一个新的线程,那好的吧,我去通知一下虚拟机(JVM),这里有个家伙想要开启一个新线程了,你有没有空,有空就帮它开启一下吧,然后虚拟机接收到这个请求后,等到空闲时,就满足了它的这个要求。

那这个时候我们就知道了,start这个方法是用来通知虚拟机,我想要开启一个新的线程,这个时候虚拟机在条件允许的情况下,就会满足它的要求。同时我们还需要注意一个点,真正发出通知的是主线程,是它来帮我们发出这个请求的。所以,正确启动线程的方式是调用Thread类的start方法。

重复调用start方法时,会发生什么

我就提前告诉你好了,会抛出一个IllegalThreadStateException的异常,至于是为什么,就需要我们去看看Thread类的源码了,我们进入到Thread类中的start方法后,会发现,在方法的一开始,就会做一个判断:

if (threadStatus != 0)
        throw new IllegalThreadStateException();

如果threadStatus != 0 的话就抛出这个异常了,threadStatus又是什么,它代表了一个线程的状态,线程有6个状态,至于是哪6个状态,后续会讲的。在Thread类初始化的时候,threadStatus就已经被赋值为0了,当我们调用了start方法之后,这个状态值就会被修改,所以,当我们重复调用start方法的时候,就会抛出异常了

以上是关于线程如何正确的启动的主要内容,如果未能解决你的问题,请参考以下文章

如何启动匿名线程类

newCacheThreadPool()newFixedThreadPool()newScheduledThreadPool()newSingleThreadExecutor()自定义线程池(代码片段

为啥基于锁的程序不能组成正确的线程安全片段?

如何正确地将多个片段添加到片段过渡?

如何在主活动中正确检索片段中的视图

Android 怎么启动一个工作线程及线程如何与UI线程交互