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è)父類,這是此方法的局限。
下面看例子:
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)線程:
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接口完成。
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)的資源):
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類完成:
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í)賣票:
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)資源共享,下面看例子:
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主線程中的:
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ì)算邏輯。
第四:在多線程編程這塊,我們經(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線程(主線程)中:
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è)例子:
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 來得緊湊。
下面看例子:
復(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ì)算邏輯。
第四:在多線程編程這塊,我們經(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 來得緊湊。
您可能感興趣的文章:
- android Handler詳細(xì)使用方法實(shí)例
- Android Handler之消息循環(huán)的深入解析
- Android Handler主線程和一般線程通信的應(yīng)用分析
- android的消息處理機(jī)制(圖文+源碼分析)—Looper/Handler/Message
- android使用handlerthread創(chuàng)建線程示例
- 詳解Android中Handler的使用方法
- Handler與Android多線程詳解
- Android定時(shí)器和Handler用法實(shí)例分析
- 詳解Android中Handler的內(nèi)部實(shí)現(xiàn)原理
- android線程消息機(jī)制之Handler詳解
相關(guān)文章
Android StrictMode運(yùn)行流程(推薦)
strictmode是android在 API9后引入的檢測影響app運(yùn)行流暢性的一種機(jī)制。這篇文章給大家介紹了android strictmode運(yùn)行流程,需要的朋友參考下吧2018-01-01Android編程實(shí)現(xiàn)AlertDialog自定義彈出對(duì)話框的方法示例
這篇文章主要介紹了Android編程實(shí)現(xiàn)AlertDialog自定義彈出對(duì)話框的方法,結(jié)合實(shí)例形式分析了Android AlertDialog自定義彈出對(duì)話框的基本功能與事件監(jiān)聽實(shí)現(xiàn)技巧,需要的朋友可以參考下2017-07-07Android 應(yīng)用更換皮膚實(shí)現(xiàn)方法
本文主要介紹Android 應(yīng)用更換皮膚,Android應(yīng)用如果想更換皮膚這里幫大家整理了相關(guān)資料,有需要的小伙伴可以參考下2016-08-08Android動(dòng)畫之小球擬合動(dòng)畫實(shí)例
這篇文章主要介紹了Android動(dòng)畫之小球擬合動(dòng)畫實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-07-07Android 仿余額寶數(shù)字跳動(dòng)動(dòng)畫效果完整代碼
這篇文章主要介紹了Android 仿余額寶數(shù)字跳動(dòng)動(dòng)畫效果完整代碼,需要的朋友可以參考下2017-11-11Flutter實(shí)現(xiàn)心動(dòng)的動(dòng)畫特效
為了追求更好的用戶體驗(yàn),有時(shí)候我們需要一個(gè)類似心跳一樣跳動(dòng)著的控件來吸引用戶的注意力。本文將利用Flutter實(shí)現(xiàn)這一動(dòng)畫特效,需要的可以參考一下2022-04-04