亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

Android開發筆記之:Handler Runnable與Thread的區別詳解

2020-04-11 12:19:53
字體:
來源:轉載
供稿:網友
在java中可有兩種方式實現多線程,一種是繼承Thread類,一種是實現Runnable接口;Thread類是在java.lang包中定義的。一個類只要繼承了Thread類同時覆寫了本類中的run()方法就可以實現多線程操作了,但是一個類只能繼承一個父類,這是此方法的局限。
下面看例子:
復制代碼 代碼如下:

package org.thread.demo;
class MyThread extends Thread{
private String name;
public MyThread(String name) {
super();
this.name = name;
}
public void run(){
for(int i=0;i<10;i++){
System.out.println("線程開始:"+this.name+",i="+i);
}
}
}
package org.thread.demo;
public class ThreadDemo01 {
public static void main(String[] args) {
MyThread mt1=new MyThread("線程a");
MyThread mt2=new MyThread("線程b");
mt1.run();
mt2.run();
}
}

但是,此時結果很有規律,先第一個對象執行,然后第二個對象執行,并沒有相互運行。在JDK的文檔中可以發現,一旦調用start()方法,則會通過JVM找到run()方法。下面啟動start()方法啟動線程:
復制代碼 代碼如下:

package org.thread.demo;  
public class ThreadDemo01 {  
public static void main(String[] args) {  
MyThread mt1=new MyThread("線程a");  
MyThread mt2=new MyThread("線程b");  
mt1.start();  
mt2.start();  
}  
}; 

這樣程序可以正常完成交互式運行。那么為啥非要使用start();方法啟動多線程呢?
在JDK的安裝路徑下,src.zip是全部的java源程序,通過此代碼找到Thread中的start()方法的定義,可以發現此方法中使用了private native void start0();其中native關鍵字表示可以調用操作系統的底層函數,那么這樣的技術成為JNI技術(java Native Interface)
Runnable接口
在實際開發中一個多線程的操作很少使用Thread類,而是通過Runnable接口完成。
復制代碼 代碼如下:

public interface Runnable{  
public void run();  

例子:
package org.runnable.demo;  
class MyThread implements Runnable{  
private String name;  
public MyThread(String name) {  
this.name = name;  
}
public void run(){  
for(int i=0;i<100;i++){  
System.out.println("線程開始:"+this.name+",i="+i);  
}  
}  
}; 

但是在使用Runnable定義的子類中沒有start()方法,只有Thread類中才有。此時觀察Thread類,有一個構造方法:public Thread(Runnable targer)此構造方法接受Runnable的子類實例,也就是說可以通過Thread類來啟動Runnable實現的多線程。(start()可以協調系統的資源):
復制代碼 代碼如下:

package org.runnable.demo;  
import org.runnable.demo.MyThread;  
public class ThreadDemo01 {  
public static void main(String[] args) {  
MyThread mt1=new MyThread("線程a");  
MyThread mt2=new MyThread("線程b");  
new Thread(mt1).start();  
new Thread(mt2).start();  
}  


兩種實現方式的區別和聯系:
在程序開發中只要是多線程肯定永遠以實現Runnable接口為主,因為實現Runnable接口相比繼承Thread類有如下好處:
•避免點繼承的局限,一個類可以繼承多個接口。
•適合于資源的共享
以賣票程序為例,通過Thread類完成:
復制代碼 代碼如下:

package org.demo.dff;  
class MyThread extends Thread{  
private int ticket=10;  
public void run(){  
for(int i=0;i<20;i++){  
if(this.ticket>0){  
System.out.println("賣票:ticket"+this.ticket--);  
}  
}  
}  
}; 

下面通過三個線程對象,同時賣票:
復制代碼 代碼如下:

package org.demo.dff;  
public class ThreadTicket {  
public static void main(String[] args) {  
MyThread mt1=new MyThread();  
MyThread mt2=new MyThread();  
MyThread mt3=new MyThread();  
mt1.start();//每個線程都各賣了10張,共賣了30張票  
mt2.start();//但實際只有10張票,每個線程都賣自己的票  
mt3.start();//沒有達到資源共享  
}  


如果用Runnable就可以實現資源共享,下面看例子:
復制代碼 代碼如下:

package org.demo.runnable;
class MyThread implements Runnable{
private int ticket=10;
public void run(){
for(int i=0;i<20;i++){
if(this.ticket>0){
System.out.println("賣票:ticket"+this.ticket--);
}
}
}
}
package org.demo.runnable;
public class RunnableTicket {
public static void main(String[] args) {
MyThread mt=new MyThread();
new Thread(mt).start();//同一個mt,但是在Thread中就不可以,如果用同一
new Thread(mt).start();//個實例化對象mt,就會出現異常
new Thread(mt).start();
}
}; 

雖然現在程序中有三個線程,但是一共賣了10張票,也就是說使用Runnable實現多線程可以達到資源共享目的。
Runnable接口和Thread之間的聯系:
public class Thread extends Object implements Runnable
發現Thread類也是Runnable接口的子類。
第二:
Thread是系統給你的資源,有了Thread你才有從CPU那里得到可執行時間片的權力, Thread并不認識你的程序,不知道有test 這樣的類,因為編序員有千千萬,每個人命名都不一樣,想要做的事都不一樣, 所以 Thread只認識一個! 那就是Runnable 。 Thread認識Runnable 并且知道Runnable 里面有一個run方法. 一旦調用Thread的start方法,Runnable 方法里的run就會被Thread自動運行。 所以,當我們把我們的類繼承(這里應該叫實現接口)自Runnable 的時候,我們的程序就是屬于Runnable 一個類型的了。 雖然是Runnable 的子類,但人家認識你爸爸,當然也知道了你。 Thread可以不管你內部有什么情況,他只管你有run()方法就行了,他就調start讓你去運行run 所以我們在run里面寫點東西,這樣就可以讓系統運行我們想要做的代碼了。 是不是很通俗很易懂呢? 所以要運行線程的步驟是, 1。生成我們自己的類對象 2。從系統那里得到Thread 3。讓Threa調我們的類對象,讓其start起來 代碼: test a=new test(); Thread thread=new Thread(a); //Thread需要一個參數,就是你編的線程類,這樣他就認識了你的線程,也有資格向系統申請拿到CPU時間片thread.start(); 你可以簡單點寫: new Thread(a).start();
第三:
Runnable 并不一定是新開一個線程,比如下面的調用方法就是運行在UI主線程中的:
復制代碼 代碼如下:

     Handler mHandler=new Handler();
     mHandler.post(new Runnable(){
        @Override public void run()
        { // TODO Auto-generated method stub
         }
     });

官方對這個方法的解釋如下,注意其中的:“The runnable will be run on the user interface thread. ”
boolean android.view.View .post(Runnable action)
Causes the Runnable to be added to the message queue. The runnable will be run on the user interface thread.
Parameters:
action The Runnable that will be executed.
Returns:
Returns true if the Runnable was successfully placed in to the message queue. Returns false on failure, usually because the looper processing the message queue is exiting.
我們可以通過調用handler的post方法,把Runnable對象(一般是Runnable的子類)傳過去;handler會在looper中調用這個Runnable的Run方法執行。
Runnable是一個接口,不是一個線程,一般線程會實現Runnable。所以如果我們使用匿名內部類是運行在UI主線程的,如果我們使用實現這個Runnable接口的線程類,則是運行在對應線程的。
具體來說,這個函數的工作原理如下:
View.post(Runnable)方法。在post(Runnable action)方法里,View獲得當前線程(即UI線程)的Handler,然后將action對象post到Handler里。在Handler里,它將傳遞過來的action對象包裝成一個Message(Message的callback為action),然后將其投入UI線程的消息循環中。在Handler再次處理該Message時,有一條分支(未解釋的那條)就是為它所設,直接調用runnable的run方法。而此時,已經路由到UI線程里,因此,我們可以毫無顧慮的來更新UI。
如下圖,前面看到的代碼,我們這里Message的callback為一個Runnable的匿名內部類
這種情況下,由于不是在新的線程中使用,所以千萬別做復雜的計算邏輯。

image


第四:在多線程編程這塊,我們經常要使用Handler,Thread和Runnable這三個類,那么他們之間的關系你是否弄清楚了呢?
首先說明Android的CPU分配的最小單元是線程,Handler一般是在某個線程里創建的,因而Handler和Thread就是相互綁定的,一一對應。
而Runnable是一個接口,Thread是Runnable的子類。所以說,他倆都算一個進程。
HandlerThread顧名思義就是可以處理消息循環的線程,他是一個擁有Looper的線程,可以處理消息循環。
與其說Handler和一個線程綁定,不如說Handler是和Looper一一對應的。
最后需要說明的是,在UI線程(主線程)中:
復制代碼 代碼如下:

  mHandler=new Handler();
  mHandler.post(new Runnable(){
  void run(){
  //執行代碼...
  }
  });
  這個線程其實是在UI線程之內運行的,并沒有新建線程。
  常見的新建線程的方法是:
  Thread thread = new Thread();
  thread.start();
  HandlerThread thread = new HandlerThread("string");
  thread.start();

第五:Java Runnable接口在進行相關編寫的時候需要我們不斷的學習相關代碼。下面我們就來看炫如何才能使用相關的代碼。Runnable接口只有一個方法run(),我們聲明自己的類實現Runnable接 口并提供這一方法,將我們的線程代碼寫入其中,就完成了這一部分的任務。
但是Runnable接口并沒有任何對線程的支持,我們還必須創建Thread類 的實例,這一點通過Thread類的構造函數public Thread(Runnable target);來實現。下面是一個例子:
復制代碼 代碼如下:

public class MyThread implements Runnable
{
int count= 1, number;
public MyThread(int num)
{
numnumber = num;
System.out.println("創建線程 " + number);
}
public void run()
{
while(true)
{
System.out.println
("線程 " + number + ":計數 " + count);
if(++count== 6) return;
}
}
public static void main(String args[])
{
for(int i = 0; i 〈 5;
i++) new Thread(new MyThread(i+1)).start();
}
}

嚴格地說,創建Thread子類的實例也是可行的,但是必須注意的是,該子類必須沒有覆蓋 Thread 類的 run 方法,否則該線程執行的將是子類的 run 方法,而不是我們用以實現Runnable 接口的類的 run 方法,對此大家不妨試驗一下。
使用 Java Runnable接口來實現多線程使得我們能夠在一個類中包容所有的代碼,有利于封裝,它的缺點在于,我們只能使用一套代碼,若想創建多個線程并使各個線程執行不同的代 碼,則仍必須額外創建類,如果這樣的話,在大多數情況下也許還不如直接用多個類分別繼承 Thread 來得緊湊。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美专区中文字幕| 亚洲电影成人av99爱色| 菠萝蜜影院一区二区免费| 都市激情亚洲色图| 日韩电视剧在线观看免费网站| 91网站在线免费观看| 在线色欧美三级视频| 欧美成人黑人xx视频免费观看| 成人h片在线播放免费网站| 91久久精品久久国产性色也91| 97精品视频在线播放| 久久久综合av| 精品免费在线观看| 欧美亚洲另类制服自拍| 亚洲性生活视频| 亚洲第一男人天堂| 欧美一级高清免费| 国产精品日本精品| 国产欧美一区二区三区视频| 日韩av电影在线免费播放| 成人动漫网站在线观看| 色老头一区二区三区在线观看| 91香蕉嫩草影院入口| 久久精品亚洲热| 亚洲一区999| 久久久免费精品视频| 久久久av网站| 国产日韩中文字幕在线| 高清亚洲成在人网站天堂| 久久亚洲综合国产精品99麻豆精品福利| 国产精品视频自拍| 国产精品成人一区| 亚洲精品乱码久久久久久按摩观| 中文字幕日韩欧美精品在线观看| 国产精品91在线观看| 91老司机精品视频| 日韩精品高清在线| 原创国产精品91| 2019中文字幕免费视频| 日韩在线欧美在线国产在线| 欧美色道久久88综合亚洲精品| 欧美三级xxx| 久久久精品999| 日韩极品精品视频免费观看| 亚洲男人天堂手机在线| 狠狠躁夜夜躁人人躁婷婷91| 亚洲精品自拍视频| 久久精品成人欧美大片| 亚洲免费av电影| 欧美国产日产韩国视频| 日韩电影大全免费观看2023年上| 正在播放欧美视频| 亚洲欧美国产一区二区三区| 亚洲专区在线视频| 欧美精品性视频| 欧美亚洲一区在线| 久久成年人免费电影| 欧美高清性猛交| 欧美高清视频在线观看| 久久大大胆人体| 欧美激情二区三区| 日韩综合视频在线观看| 亚洲国产欧美在线成人app| 2019最新中文字幕| 精品国产电影一区| 国产精品麻豆va在线播放| 亚洲欧洲激情在线| 欧美国产日韩一区二区在线观看| 色伦专区97中文字幕| 国产精品中文字幕久久久| 日韩中文字幕在线视频播放| 久久综合免费视频| 日韩免费看的电影电视剧大全| 国产xxx69麻豆国语对白| 97免费视频在线| 国产精品久久av| 日韩视频在线免费观看| 国产va免费精品高清在线观看| 亚洲在线第一页| www.欧美精品| 欧美高清视频在线播放| 777午夜精品福利在线观看| 欧美日韩在线免费| 国产精品久久婷婷六月丁香| 在线电影欧美日韩一区二区私密| 91精品国产综合久久香蕉| 日韩成人网免费视频| 国产精品自拍偷拍| 国模私拍一区二区三区| 亚洲国产精品人久久电影| 亚洲成人a级网| 国产欧美精品一区二区三区介绍| 九九九热精品免费视频观看网站| 欧美丰满片xxx777| 国产在线观看91精品一区| 亚洲精品久久久久中文字幕欢迎你| 日韩中文视频免费在线观看| 最新国产精品拍自在线播放| 久久91超碰青草是什么| 久久99视频免费| 97视频免费在线看| 激情亚洲一区二区三区四区| 原创国产精品91| 亚洲精品国偷自产在线99热| 狠狠久久五月精品中文字幕| 国产aaa精品| 亚洲成人精品在线| 亚洲第一中文字幕| 日韩最新中文字幕电影免费看| 国产精品露脸av在线| 2025国产精品视频| 国产91精品久久久久久久| 国产亚洲精品一区二555| 色综合91久久精品中文字幕| 日韩中文在线视频| 国产精品免费福利| 456亚洲影院| 久久免费精品视频| 久久久久久久久久久人体| 久久精品99久久香蕉国产色戒| 日韩电影免费观看在线| 青青草国产精品一区二区| 亚洲成人久久网| 98精品在线视频| 91国产中文字幕| 久久久久成人精品| 成人网在线观看| 国产欧美一区二区三区久久| 91国自产精品中文字幕亚洲| 国产美女扒开尿口久久久| 亚洲精品日韩激情在线电影| 日韩av片电影专区| 国产美女精品免费电影| 亚洲三级免费看| 91九色视频在线| 91精品国产91久久久久福利| 日韩精品极品视频| 精品中文字幕在线2019| 日韩精品在线观看一区二区| 亚洲午夜女主播在线直播| 亚洲天天在线日亚洲洲精| 亚洲精品日韩丝袜精品| 国产精品美乳一区二区免费| 欧美有码在线观看| 欧美激情欧美狂野欧美精品| 91精品国产91久久久| 91沈先生作品| 欧美国产精品va在线观看| 一本色道久久88综合日韩精品| 久久九九亚洲综合| 亚洲男女自偷自拍图片另类| 亚洲剧情一区二区| 欧美性jizz18性欧美| 成人啪啪免费看| 亚洲精品小视频在线观看| 国产精品久久久久国产a级| 8050国产精品久久久久久| 亚洲国产成人精品一区二区| 一本色道久久综合狠狠躁篇的优点| 亚洲一区二区三区sesese| 中文字幕亚洲激情| 日本中文字幕久久看| 久久久久在线观看| 韩剧1988在线观看免费完整版|