概要
Thread类包含start()和run()方法,它们的区别是什么?本章将对此作出解答。
start() 和 run()的区别说明
start() :
它的作用是启动一个新线程,新线程会执行相应的run()方法。start()不能被重复调用。
run() :
run()就和普通的成员方法一样,可以被重复调用。单独调用run()的话,会在当前线程中执行run(),而并不会启动新线程!
下面以代码来进行说明。
class MyThread extends Thread{
public void run(){
...
}
};
MyThread mythread = new MyThread();
mythread.start()会启动一个新线程,并在新线程中运行run()方法。
而mythread.run()则会直接在当前线程中运行run()方法,并不会启动一个新线程来运行run()。
start() 和 run()的区别示例
下面,通过一个简单示例演示它们之间的区别。源码如下:
示例
class MyThread extends Thread {
public MyThread(String name) {
super(name);
}
@Override
public void run() {
System.out.println(" " + Thread.currentThread().getName() + " is running");
}
};
public class Demo {
public static void main(String[] args) {
Thread mythread = new MyThread("mythread");
System.out.println(Thread.currentThread().getName() + " call mythread.run()");
mythread.run();
System.out.println(Thread.currentThread().getName() + " call mythread.start()");
mythread.start();
}
}
运行结果
main call mythread.run()
main is running
main call mythread.start()
mythread is running
结果说明
(01)、Thread.currentThread().getName()是用于获取“当前线程”的名字。当前线程是指正在cpu中调度执行的线程。
(02)、mythread.run()是在“主线程main”中调用的,该run()方法直接运行在“主线程main”上。
(03)、mythread.start()会启动“线程mythread”,“线程mythread”启动之后,会调用run()方法;此时的run()方法是运行在“线程mythread”上。
start() 和 run()相关源码(基于JDK1.7+)
Thread.java中start()方法的源码如下:
public synchronized void start() {
// 如果线程不是"就绪状态",则抛出异常!(新创建一个线程默认为0)
if (threadStatus != 0)
throw new IllegalThreadStateException();
// 将线程添加到ThreadGroup中
group.add(this);
boolean started = false;
try {
// 通过start0()启动线程
start0();
// 设置started标记
started = true;
} finally {
try {
if (!started) {
group.threadStartFailed(this);
}
} catch (Throwable ignore) {
}
}
}
说明:
start()实际上是通过本地方法start0()启动线程的。而start0()会新运行一个线程,新线程会调用run()方法。
private native void start0();
Thread.java中run()的代码如下:
public void run() {
if (target != null) {
target.run();
}
}
说明:
target是一个Runnable对象。run()就是直接调用Thread线程的Runnable成员的run()方法,并不会新建一个线程。
疑问标记
—— 以下的代码我觉得有疑问:通过debug,并没有看到target 被初始化了,而且也没有进入run()方法中,有知道的请解答 ———
Thread.java中run()的代码如下:
public void run() {
if (target != null) {
target.run();
}
}
疑问解答
原谅我的小白失误,run方法被我们自己给重载了,当然不会进入父类的run方法了。也就和target 没有关系了,而start0()又是一个原生的方法。
如果是以下方式开启线程,则会调用tun方法。在构造中给target给初始化了
new Thread(new Runnable(){
@Override
public void run() {
System.out.println(" " + Thread.currentThread().getName() + " is running");
}
}).start();