Java多線程基礎--03之 Thread中start()和run()的區別

0. 本文目錄

Thread類包含start()和run()方法,它們的區別是什麼?本章將對此作出解答。

本章內容包括:

1. 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()。

2. start() 和 run()的區別示例

下面,通過一個簡單示例演示它們之間的區別。源碼如下:

// Demo.java 的源碼
class MyThread extends Thread{  
    public MyThread(String name) {
        super(name);
    }

    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

結果說明:

  • Thread.currentThread().getName()是用於獲取“當前線程”的名字。當前線程是指正在cpu中調度執行的線程。
  • mythread.run()是在“主線程main”中調用的,該run()方法直接運行在“主線程main”上。
  • mythread.start()會啓動“線程mythread”,“線程mythread”啓動之後,會調用run()方法;此時的run()方法是運行在“線程mythread”上。

3.start() 和 run()相關源碼(基於JDK1.7.0_40)

Thread.java中start()方法的源碼如下:

public synchronized void start() {
    // 如果線程不是"就緒狀態",則拋出異常!
    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()方法,並不會新建一個線程。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章