欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android開發(fā)筆記之:Handler Runnable與Thread的區(qū)別詳解

 更新時(shí)間:2013年05月28日 10:20:28   作者:  
本篇文章是對(duì)在Android中Handler Runnable與Thread的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
在java中可有兩種方式實(shí)現(xiàn)多線程,一種是繼承Thread類,一種是實(shí)現(xiàn)Runnable接口;Thread類是在java.lang包中定義的。一個(gè)類只要繼承了Thread類同時(shí)覆寫了本類中的run()方法就可以實(shí)現(xiàn)多線程操作了,但是一個(gè)類只能繼承一個(gè)父類,這是此方法的局限。
下面看例子:
復(fù)制代碼 代碼如下:

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

但是,此時(shí)結(jié)果很有規(guī)律,先第一個(gè)對(duì)象執(zhí)行,然后第二個(gè)對(duì)象執(zhí)行,并沒有相互運(yùn)行。在JDK的文檔中可以發(fā)現(xiàn),一旦調(diào)用start()方法,則會(huì)通過JVM找到run()方法。下面啟動(dòng)start()方法啟動(dòng)線程:
復(fù)制代碼 代碼如下:

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

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

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類中才有。此時(shí)觀察Thread類,有一個(gè)構(gòu)造方法:public Thread(Runnable targer)此構(gòu)造方法接受Runnable的子類實(shí)例,也就是說可以通過Thread類來啟動(dòng)Runnable實(shí)現(xiàn)的多線程。(start()可以協(xié)調(diào)系統(tǒng)的資源):
復(fù)制代碼 代碼如下:

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


兩種實(shí)現(xiàn)方式的區(qū)別和聯(lián)系:
在程序開發(fā)中只要是多線程肯定永遠(yuǎn)以實(shí)現(xiàn)Runnable接口為主,因?yàn)閷?shí)現(xiàn)Runnable接口相比繼承Thread類有如下好處:
•避免點(diǎn)繼承的局限,一個(gè)類可以繼承多個(gè)接口。
•適合于資源的共享
以賣票程序?yàn)槔ㄟ^Thread類完成:
復(fù)制代碼 代碼如下:

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

下面通過三個(gè)線程對(duì)象,同時(shí)賣票:
復(fù)制代碼 代碼如下:

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();//每個(gè)線程都各賣了10張,共賣了30張票  
mt2.start();//但實(shí)際只有10張票,每個(gè)線程都賣自己的票  
mt3.start();//沒有達(dá)到資源共享  
}  


如果用Runnable就可以實(shí)現(xiàn)資源共享,下面看例子:
復(fù)制代碼 代碼如下:

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();//同一個(gè)mt,但是在Thread中就不可以,如果用同一
new Thread(mt).start();//個(gè)實(shí)例化對(duì)象mt,就會(huì)出現(xiàn)異常
new Thread(mt).start();
}
}; 

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

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

官方對(duì)這個(gè)方法的解釋如下,注意其中的:“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.
我們可以通過調(diào)用handler的post方法,把Runnable對(duì)象(一般是Runnable的子類)傳過去;handler會(huì)在looper中調(diào)用這個(gè)Runnable的Run方法執(zhí)行。
Runnable是一個(gè)接口,不是一個(gè)線程,一般線程會(huì)實(shí)現(xiàn)Runnable。所以如果我們使用匿名內(nèi)部類是運(yùn)行在UI主線程的,如果我們使用實(shí)現(xiàn)這個(gè)Runnable接口的線程類,則是運(yùn)行在對(duì)應(yīng)線程的。
具體來說,這個(gè)函數(shù)的工作原理如下:
View.post(Runnable)方法。在post(Runnable action)方法里,View獲得當(dāng)前線程(即UI線程)的Handler,然后將action對(duì)象post到Handler里。在Handler里,它將傳遞過來的action對(duì)象包裝成一個(gè)Message(Message的callback為action),然后將其投入U(xiǎn)I線程的消息循環(huán)中。在Handler再次處理該Message時(shí),有一條分支(未解釋的那條)就是為它所設(shè),直接調(diào)用runnable的run方法。而此時(shí),已經(jīng)路由到UI線程里,因此,我們可以毫無顧慮的來更新UI。
如下圖,前面看到的代碼,我們這里Message的callback為一個(gè)Runnable的匿名內(nèi)部類
這種情況下,由于不是在新的線程中使用,所以千萬別做復(fù)雜的計(jì)算邏輯。

image


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

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

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

public class MyThread implements Runnable
{
int count= 1, number;
public MyThread(int num)
{
numnumber = num;
System.out.println("創(chuàng)建線程 " + number);
}
public void run()
{
while(true)
{
System.out.println
("線程 " + number + ":計(jì)數(shù) " + 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();
}
}

嚴(yán)格地說,創(chuàng)建Thread子類的實(shí)例也是可行的,但是必須注意的是,該子類必須沒有覆蓋 Thread 類的 run 方法,否則該線程執(zhí)行的將是子類的 run 方法,而不是我們用以實(shí)現(xiàn)Runnable 接口的類的 run 方法,對(duì)此大家不妨試驗(yàn)一下。
使用 Java Runnable接口來實(shí)現(xiàn)多線程使得我們能夠在一個(gè)類中包容所有的代碼,有利于封裝,它的缺點(diǎn)在于,我們只能使用一套代碼,若想創(chuàng)建多個(gè)線程并使各個(gè)線程執(zhí)行不同的代 碼,則仍必須額外創(chuàng)建類,如果這樣的話,在大多數(shù)情況下也許還不如直接用多個(gè)類分別繼承 Thread 來得緊湊。

相關(guān)文章

  • Flutter組件適配方法實(shí)現(xiàn)詳解

    Flutter組件適配方法實(shí)現(xiàn)詳解

    在寫flutter的代碼的時(shí)候?yàn)榱俗尳M件大小適配屏幕使用了flutter_screenUtil插件。在調(diào)試的時(shí)候是正常的,可以適配屏幕。但是打包之后就會(huì)失去效果,這篇文章主要介紹了Flutter組件適配方法實(shí)現(xiàn)
    2022-10-10
  • Android StrictMode運(yùn)行流程(推薦)

    Android StrictMode運(yùn)行流程(推薦)

    strictmode是android在 API9后引入的檢測影響app運(yùn)行流暢性的一種機(jī)制。這篇文章給大家介紹了android strictmode運(yùn)行流程,需要的朋友參考下吧
    2018-01-01
  • Android編程實(shí)現(xiàn)AlertDialog自定義彈出對(duì)話框的方法示例

    Android編程實(shí)現(xiàn)AlertDialog自定義彈出對(duì)話框的方法示例

    這篇文章主要介紹了Android編程實(shí)現(xiàn)AlertDialog自定義彈出對(duì)話框的方法,結(jié)合實(shí)例形式分析了Android AlertDialog自定義彈出對(duì)話框的基本功能與事件監(jiān)聽實(shí)現(xiàn)技巧,需要的朋友可以參考下
    2017-07-07
  • Flutter渲染原理深入解析

    Flutter渲染原理深入解析

    眾所周知?Flutter是由Google推出的開源的高性能跨平臺(tái)框架,一個(gè)2D渲染引擎。在Flutter中,Widget是Flutter用戶界面的基本構(gòu)成單元,可以說一切皆Widget。下面來看下Flutter框架的整體結(jié)構(gòu)組成
    2023-04-04
  • Android 應(yīng)用更換皮膚實(shí)現(xiàn)方法

    Android 應(yīng)用更換皮膚實(shí)現(xiàn)方法

    本文主要介紹Android 應(yīng)用更換皮膚,Android應(yīng)用如果想更換皮膚這里幫大家整理了相關(guān)資料,有需要的小伙伴可以參考下
    2016-08-08
  • Android動(dòng)畫之小球擬合動(dòng)畫實(shí)例

    Android動(dòng)畫之小球擬合動(dòng)畫實(shí)例

    這篇文章主要介紹了Android動(dòng)畫之小球擬合動(dòng)畫實(shí)例的相關(guān)資料,需要的朋友可以參考下
    2017-07-07
  • Android自定義帶圓角的ImageView

    Android自定義帶圓角的ImageView

    這篇文章主要為大家詳細(xì)介紹了Android自定義帶圓角的ImageView,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-08-08
  • Android Studio安裝配置方法圖文教程

    Android Studio安裝配置方法圖文教程

    這篇文章主要為大家詳細(xì)介紹了Android Studio安裝配置方法圖文教程,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-04-04
  • Android 仿余額寶數(shù)字跳動(dòng)動(dòng)畫效果完整代碼

    Android 仿余額寶數(shù)字跳動(dòng)動(dòng)畫效果完整代碼

    這篇文章主要介紹了Android 仿余額寶數(shù)字跳動(dòng)動(dòng)畫效果完整代碼,需要的朋友可以參考下
    2017-11-11
  • Flutter實(shí)現(xiàn)心動(dòng)的動(dòng)畫特效

    Flutter實(shí)現(xiàn)心動(dòng)的動(dòng)畫特效

    為了追求更好的用戶體驗(yàn),有時(shí)候我們需要一個(gè)類似心跳一樣跳動(dòng)著的控件來吸引用戶的注意力。本文將利用Flutter實(shí)現(xiàn)這一動(dòng)畫特效,需要的可以參考一下
    2022-04-04

最新評(píng)論