Java多线程系列--“基础篇”03之 Thread中start()和run()的区别

Java多线程系列-目录


概要

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();

本文From:http://www.cnblogs.com/skywang12345/p/3479083.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值