怎样才是正确的线程启动方式?

149 阅读3分钟

图片.png 一起养成写作习惯!这是我参与「掘金日新计划 · 4 月更文挑战」的第7天,点击查看活动详情

1、start()和run()的比较

代码演示:

package threadcoreknowledge.startthread;

/**
 * 描述:   对比start和run两种启动线程的方式
 */
public class StartAndRunMethod {
    public static void main(String[] args) {
        Runnable runnable = ()->{
            System.out.println(Thread.currentThread().getName());
        };
        runnable.run();

        new Thread(runnable).start();
    }
}

运行结果:

2、start()方法原理解读

2.1、start()方法含义

1、启动新线程:

  • 通知jvm,在有空闲的时候启动新线程;
  • 调用start()方法的顺序并不能决定线程启动的顺序;
  • start()方法其实会让两个线程同时运行,第一个就是我们的主线程,因为我们必须要有一个主线程或者其他的线程来执行start()方法,第二个才是我们刚刚被创建的子线程。new Thread().start()被我们的主线程或者父线程执行之后,之后去创建了我们的子线程。

2、准备工作:

  • 首先让自己处于就绪状态,这个就绪指的是已经获得了除CPU以外的其他资源,比如说已经设置了上下文,栈 ,线程状态,以及程序计数器(PC)。

3、不能重复start()

  • 线程一旦开始执行,线程状态就从最开始的new状态进入到后序的那些状态,比如说RUNNABLE。一旦线程执行完毕,就会变成终止状态,而终止状态永远都不会返回回去,所以才会抛出非法的线程状态异常

代码演示:

package threadcoreknowledge.startthread;
/**
 *  描述:演示不能两次调用start方法,否则会报错
 */
public class CanStartTwice {
    public static void main(String[] args) {
        Thread thread = new Thread();
        thread.start();
        thread.start();
    }
}

运行结果:

2.2、start()源码解析

1、启动新线程检查线程状态。

  • Java线程状态最开始被初始化为0,表示还没有启动;
  • 在start()方法的最开始要首先检查threadStatus是不是我们刚刚初始化还没有启动的状态, 如果不是,说明有问题抛出异常。

2、加入线程组。

3、调用start0()。

start方法源码:

public synchronized void start() {  
	if (threadStatus != 0)  //检查threadStatus
		throw new IllegalThreadStateException();
	group.add(this);        //加入线程组
	boolean started = false;
	try {
		start0();           //调用start0()
		started = true;
	} finally {
		try {
			if (!started) {
				group.threadStartFailed(this);
			}
		} catch (Throwable ignore) {

		}
	}
}

3、run()方法原理解读

源码解析

@Override
public void run() {
	if (target != null) {
		target.run();
	}
}

两种情况:

  • 重写Thread类的run()方法;
  • 传入target对象;

因此如果直接执行Runnable对象的run()方法,相当于执行一个普通方法,并没有起到多线程效果。所以要想真正的启动线程,不能直接调用run()方法,而是调用start()方法来间接的调用run()方法。

4、启动线程——常见面试问题

1、一个线程 两次 调用 start() 方法会出现什么情况?为什么?

  • 会抛出java.lang.IllegalThreadStateException(非法的线程状态异常)。
  • 在start()方法的最开始要首先检查线程状态是不是我们刚刚初始化还没有启动的状态, 如果不是,说明有问题抛出异常。
  • 并且线程一旦开始执行,线程状态就从最开始的new状态进入到后序的那些状态,比如说RUNNABLE。一旦线程执行完毕,就会变成终止状态,而终止状态永远都不会返回回去,所以才会抛出非法的线程状态异常

2、既然start()方法会调用run()方法,为什么我们选择调用start()方法,而不是 直接调用run() 方法呢?

  • 调用start()方法才是真正意义上启动了一个线程,它会去经历线程的各个生命周期。如果我们直接调用run()方法,它就是一个普通的方法而已,也不会用子线程去调用。