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

图片.png

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()方法,它就是一个普通的方法而已,也不会用子线程去调用。
  • 13
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 18
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 18
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

林小鹿@

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值