亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

java - 為什么要將Runnable接口的子類對象傳遞給Thread的構造函數?

瀏覽:124日期:2023-11-13 10:00:22

問題描述

此外,runnable相比thread除了繼承方面,代碼和數據獨立體現在哪?像有些博客上寫的thread不能共享資源,runnable能共享資源,將thread中的變量改成static不就行了吧?就像下面這篇說的http://blog.csdn.net/uudou/ar...

問題解答

回答1:

跟數據似乎關系不大,我覺得Runnable有兩個好處:

實現Runnable以后,即可以開個線程跑(一般是用executorService.exec(command),挫一點也可以用new Thread(command).start()),也可以不開線程阻塞式的跑(直接調用command.run());

Java 1.8以后可以用Lambda來跑,例如:

new Thread(() -> { // Do something}).start();回答2:

Runnable的好處是各種場景都可以用,比如你可以讓任何一個Class implements Runnable,但是extends Thread就有一些限制,因為Java單繼承的原因,在有些場景下沒法用。

回答3:

回答:

這個問題算是一個設計問題。

之所以將 Thread 和 Runnable 分開,是希望把線程的 '創建過程' 與線程的 '執行邏輯' 徹底分開。

也就是說:線程的創建過程是“代碼”;線程的執行邏輯是“數據”;

這聽起來有點叫人暈呼,不都是 JAVA 代碼么?怎么代碼又變成數據了呢?

我們不在這些概念上糾纏,我覺得可以倒轉過來思考這個問題,舉個例子來說明問題。

討論過程:

例如我要設計一個單線程程序,這個單線程需要完成兩個任務:

1、打印一句 hello world;2、計算一下 int a 與 int b 兩個數的和并輸出;

注意:到底是執行 1? 還是 2?是由參數 n 來決定的,n 是一個隨機數……

為了讓這兩個任務在同一個線程里執行,我們可以寫這樣的代碼:

float n = (float)Math.random();int a = 1;int b = 1;Thread t = new Thread() { @Override public void start() {if (n >= 0.5f) { System.out.println('hello world');} else { System.out.println(a + b);} }};t.start();

上面的代碼確實是可以完成任務的,但問題是我們把線程的 '創建過程' 和 '業務邏輯' 混淆在一起了……

這樣不太妙。順便說一句,從操作系統層面來看,線程的創建過程其實是非常復雜的!

Java 語言把這種復雜性都封裝得看不見了,雖然代碼上就是一個 Thread 類,調用起來似乎也沒什么門檻,但 Thread 的創建過程還是很復雜、很消耗資源的。

言歸正傳,現在我再次加入一個小小的需求,除了前面的 1、2,我再加入一個 3,顯示一下系統當前時間戳。

于是任務變成了:1、打印一句 hello world;2、計算一下 int a 與 int b 兩個數的和并輸出;3、顯示一下系統當前時間戳;

注意,這時候我們需要修改 Thread 的創建過程,也就是修改 start 函數:

float n = (float)Math.random();int a = 1;int b = 1;Thread t = new Thread() { @Override public void start() {if (n >= 0.33f) { System.out.println('hello world');} else if (n >= 0.66f) { System.out.println(a + b);} else { System.out.println(System.currentTimeMillis()); // 這里是新增的語句} }};t.start();

討論至此,讓我們仔細觀察觀察……其實:

Thread t = new Thread() { @Override public void start() {// ... }}

這部分代碼是不變的,只有 start 函數里面的代碼是隨著需求變化而修改的。

那么我們可不可以把這部分變化的內容包裝成一個接口??

這應該是一個不錯的主意!

Thread t = new Thread() { private Runnable runnable; // 這里定義一個 Runnable 類型的成員 @Override public void start() {if (null != this.runnable) { runnable.run(); // 在這里用接口來把頻繁變化的業務邏輯從線程代碼里給拿出去,只調用 run 函數} }}

到這里不知道你是否已經完全明白了? :D

哈哈,Java 的 Thread 類不是剛好提供了一個帶有 Runnable 參數的構造器么?

我們將業務代碼被放到 Runnable 接口的實現類里:

class BizLogicRun implements Runnable { @Override public void run() {float n = (float)Math.rand();int a = 1;int b = 1;if (n >= 0.33f) { System.out.println('hello world');} else if (n >= 0.66f) { System.out.println(a + b);} else { System.out.println(System.currentTimeMillis()); // 這里是新增的語句} }}

那么最后,我們可以這么調用:

Thread t = new Thread(new BizLogicRun());t.start();

這樣就完成了線程的 '創建過程' 和 '業務邏輯' 徹底拆分!這種 '拆分' 也為 Java 線程池(Thread Pool)技術做好了鋪墊。

說實話,示例代碼中的 Thread t = new Thread() { ... } 這個還是夠簡單的,但在線程池中創建 Thread 可就沒這么簡單了。

所以 '拆分' 是非常有必要的!

另外,我們是否可以想象:

class PoolRun implements Runnable { List<Runnable> runnableList;}

如果 Runable 實現類里面,夾帶的還是一個 Runnable 列表會怎么樣呢?

總結:

1、使用 Runnable 接口的目的是把線程的 '創建過程' 與線程的 '執行邏輯' 徹底分開;2、Thread 不能共享資源,Runnable 能共享資源,這個說法是不正確的;3、在討論過程中我們是從具體到抽象;4、我在例子中給出的代碼確實比較簡單,但希望能說明白問題;

好了,以上就是我對這個問題的回答,希望對你有所幫助。

標簽: java
相關文章:
主站蜘蛛池模板: 成年女人毛片免费视频 | 久久亚洲国产最新网站 | 韩国一级特黄毛片大 | 国产三级做爰高清视频a | 免费视频成人国产精品网站 | 手机在线看片日韩 | 久久成人精品视频 | 伦理片一区 | 夜色www| 亚洲 欧美 另类 综合 日韩 | 国产成 人 综合 亚洲绿色 | 91女神在线观看 | 国产精品欧美亚洲区 | 成人欧美日本免费观看 | 国产色在线 | 亚洲 国产色在线com | 久久人人精品 | 亚洲夜色夜色综合网站 | 在线一区二区三区 | 又黄又爽的视频 | 黄色一级片在线播放 | 亚洲一区二区久久 | 亚洲夂夂婷婷色拍ww47 | 99久久99久久精品免费看蜜桃 | 自偷自偷自亚洲永久 | 国产免费一区二区三区在线观看 | 国产成人精品综合久久久 | 国产日韩欧美在线一区二区三区 | 啪啪网站色大全免费 | a级欧美| 欧美性生活视频 | 网友偷自拍原创区 | 亚洲乱人伦在线 | 青青综合 | 精品一区二区在线欧美日韩 | 国产a级三级三级三级中国 国产a级午夜毛片 | 这里只有精品99re在线 | 玖玖99视频 | 青青草国产97免久久费观看 | 免费观看欧美精品成人毛片能看的 | 黄网在线观看网址入口 | 午夜婷婷网 |