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

首頁 > 系統 > Android > 正文

android 多線程技術應用

2020-04-11 12:40:31
字體:
來源:轉載
供稿:網友
多線程案例――計時器
這個案例中,屏幕啟動之后,進入如圖所示的界面。
屏幕上有一個文本框用于顯示逝去的時間,此外還有一個“停止計時”按鈕。案例的用例圖如圖所示。
 
能夠在屏幕上“實時地顯示”時間的流逝,單線程程序是無法實現的,必須要多線程程序才可以實現,即便有些計算機語言可以通過封裝好的類實現這一功能,但從本質上講這些封裝好的類就是封裝了一個線程。
綜上所述,完成本案例用到的知識及技術如下:
  1)進程和線程的概念;
  2)Java中的線程,在Java中創建線程的方式;
  3)Android中的線程,包括:Message、Handler、Looper和HandlerThread等概念。
線程究竟是什么?在Windows操作系統出現之前,個人計算機上的操作系統都是單任務系統,只有在大型計算機上才具有多任務和分時設計。Windows、Linux操作系統的出現,把原本只在大型計算機才具有的優點,帶到了個人計算機系統中。
進程概念
  一般可以在同一時間內執行多個程序的操作系統都有進程的概念。一個進程就是一個執行中的程序,而每一個進程都有自己獨立的一塊內存空間、一組系統資源。在進程的概念中,每一個進程的內部數據和狀態都是完全獨立的。在Windows操作系統下我們可以通過〈Ctrl+Alt+Del〉組合鍵查看進程,在UNIX和Linux操作系統下是通過PS命令查看進程的。打開Windows當前運行的進程,如圖所示。
 
在Windows操作系統中一個進程就是一個exe或dll程序,它們相互獨立,互相也可以通信,在Android操作系統中進程間的通信應用也是很多的。
線程概念
  多線程指的是在單個程序中可以同時運行多個不同的線程,執行不同的任務。多線程意味著一個程序的多行語句可以看上去幾乎在同一時間內同時運行。
  線程與進程相似,是一段完成某個特定功能的代碼,是程序中單個順序的流控制。但與進程不同的是,同類的多個線程共享一塊內存空間和一組系統資源,所以系統在各個線程之間切換時,資源占用要比進程小得多,正因如此,線程也被稱為輕量級進程。一個進程中可以包含多個線程。圖所示是計時器程序進程和線程之間的關系,主線程負責管理子線程,即子線程的啟動、掛起、停止等操作。
 
Java中的線程
  Java的線程類是java.lang.Thread類。當生成一個Thread類的對象之后,一個新的線程就產生了。Java中每個線程都是通過某個特定Thread對象的方法run()來完成其操作的,方法run( )稱為線程體。
  下面是構建線程類幾種常用的方法:
  public Thread()
  public Thread(Runnable target)
  public Thread(Runnable target, String name)
  public Thread(String name)
  參數target是一個實現Runnable接口的實例,它的作用是實現線程體的run()方法。目標target可為null,表示由本身實例來執行線程。name參數指定線程名字,但沒有指定的構造方法,線程的名字是JVM分配的,例如JVM指定為thread-1、thread-2等名字。
1、Java中的實現線程體方式1
  在Java中有兩種方法實現線程體:一是繼承線程類Thread,二是實現接口Runnable。下面我們先看看繼承線程類Thread方式。
如果采用第1種方式,它繼承線程類Thread并重寫其中的方法 run(),在初始化這個類實例的時候,目標target可為null,表示由本實例來執行線程體。由于Java只支持單重繼承,用這種方法定義的類不能再繼承其他父類,例如代碼清單如圖:
復制代碼 代碼如下:

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
public class textThread extends Thread {
boolean flag = true;
int timer = 0;
@Override
public void run() {
super.run();
try {
while (flag) {
this.currentThread().sleep(1000);
timer++;
System.out.print(timer);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
textThread thread = new textThread();
thread.start();
System.out.print("啟動計時器...");
BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
try {
String line = reader.readLine();
if(line.equalsIgnoreCase("1")){
//thread.stop();
thread.flag = false;
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}

在main主方法中通過new textThread()創建子線程,并通過thread.start()方法啟動子線程,main主方法所在線程為主線程,主線程負責管理其他的子線程。本例進程、主線程和子線程之間的關系如圖所示。
  子線程啟動之后就開始調用run()方法,run()是一個線程體,我們在子線程中處理事情就是在這里編寫代碼實現的。本案例中子線程要做的事情就是:休眠1s,計時器加1,再反復執行。Thread.currentThread().sleep(1000)就是休眠1s。
  為了能夠停止線程,我們在主線程中增加了一個標識,通過在控制臺輸入一個字符
  “1”來改變該標識t1.isRunning = false,從而結束這個線程。
 
注意
  事實上線程中有一個stop()方法也可以停止線程,但是由于這種方法會產生線程死鎖問題,所以在新版JDK中已經廢止了,它的替代解決方法就是增加標識,就是我們在本例中采用的方案。
  很多人覺得線程難理解,主要有兩個問題
  線程休眠,既然線程已經休眠了,程序的運行速度還能提高嗎?
  線程體一般都進行死循環,既然線程死循環,程序就應該死掉了,就會沒有反應。
  1.關于線程休眠問題
  對線程休眠問題頭痛的讀者,其實還是在用單線程的思維模式考慮問題,多數情況下我們的PC都是單CPU的,某個時間點只能有一個線程運行。所謂多線程就是多個線程交替執行就好像同時運行似的。因此,休眠當前線程可以交出CPU控制權,讓其他的線程有機會運行,多個線程之間只有交替運行效率才是最高的,這就像我們開車過十字路口,只有我等等,讓你先過,你再等等讓他先過,才能保證最高效率,否則就會造成交通系統崩潰,對線程情況也是一樣的。因此,多線程中線程的休眠是程序運行的最有效方式。
  2.關于線程體死循環問題
  在單線程中如果是死循環,程序應就會死掉,沒有反應,但是多線程中線程體(run方法)中的死循環,可以保證線程一直運行,如果不循環線程,則運行一次就停止了。在上面的例子中線程體運行死循環,可以保證線程一直運行,每次運行都休眠1s,然后喚醒,再然后把時間信息輸出到控制臺。所以,線程體死循環是保證子線程一直運行的前提。由于是子線程它不會堵塞主線程,就不會感覺到程序死掉了。但是需要注意的是有時我們確實執行一次線程體,就不需要循環了。
 程序運行后開始啟動線程,線程啟動后就計算逝去的時間,每過1s將結果輸出到控制臺。當輸入1字符后線程停止,程序終止。如圖所示。
 
Java中的實現線程體方式2
  上面介紹繼承Thread方式實現線程體,下面介紹另一種方式,這種方式是提供一個實現接口Runnable的類作為一個線程的目標對象,構造線程時有兩個帶有Runnable target參數的構造方法:
  Thread(Runnable target);
  Thread(Runnable target, String name)。
  其中的target就是線程目標對象了,它是一個實現Runnable的類,在構造Thread類時候把目標對象(實現Runnable的類)傳遞給這個線程實例,由該目標對象(實現Runnable的類)提供線程體run()方法。這時候實現接口Runnable的類仍然可以繼承其他父類。
請參看代碼清單,這是一個Java AWT的窗體應用程序。
復制代碼 代碼如下:

import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
public class textThread2 extends Frame implements Runnable, ActionListener {
public Label label;
public Button button;
public Thread ClockThread;
public boolean isRunning;
public int timer = 0;
public textThread2() {
button = new Button("停止計時器");
label = new Label("計算器啟動");
button.addActionListener(this);
setLayout(new BorderLayout());
add(button, "North");
add(label, "Center");
setSize(320, 480);
setVisible(true);
// textThread2 textThread2 = new textThread2();
ClockThread = new Thread(this);
ClockThread.start();
isRunning = true;
}
@Override
public void actionPerformed(ActionEvent e) {
isRunning = false;
}
@Override
public void run() {
// TODO Auto-generated method stub
try {
while (isRunning) {
Thread.currentThread().sleep(1000);
timer++;
label.setText("逝去了" + timer);
}
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
/**
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
textThread2 textThread2 = new textThread2();
}
}

其中關于Java AWT知識就不在這里介紹了,有興趣的讀者可以自己看看相關書籍。在本例中構建AWT窗體的應用程序方式是繼承Frame類。采用第1種方式――繼承方式實現線程體是不可以的,因為Java是單繼承的,這個類不能既繼承Frame又繼承Thread。應該采用第2種方式――實現Runnable接口方式。Runnable接口也有一個run()方法,它是實現線程體方法,其代碼處理與上一節是一樣。需要注意的是,在第2種方法中,創建了一個Thread成員變量clockThread,才用構造方法new Thread(this)創建一個線程對象,其中創建線程使用的構造方法是Thread(Runnable target),其中的this就是代表本實例,它是一個實現了Runnable接口的實現類。
程序運行結果如圖所示,屏幕開始加載的時候線程啟動開始計算時間,1s更新一次UI,當單擊“結束計時”按鈕時,停止計時。
 
Java中的實現線程體方式3
  實現線程體方式3是實現線程體方式2的變種,本質上還是實現線程體方式2,但是在Android應用開發中經常采用第3種方式。下面我們看第3種方式的計時器代碼清單.
復制代碼 代碼如下:

import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
public class TextThread3 extends Frame implements ActionListener {
private Button button;
private Label label;
private Thread clockThread;
private int timer = 0;
private boolean isRunning = true;
public TextThread3() {
button = new Button("停止計時");
label = new Label("計時器開始。。。");
button.addActionListener(this);
setLayout(new BorderLayout());
add(button, "North");
add(label, "Center");
setSize(320, 480);
setVisible(true);
clockThread = new Thread(new Runnable() {
@Override
public void run() {
while (isRunning) {
try {
Thread.currentThread().sleep(1000);
timer ++;
label.setText("逝去了:"+timer);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
});
clockThread.start();
isRunning = true;
}
@Override
public void actionPerformed(ActionEvent e) {
isRunning = false;
}
/**
* @param args
*/
public static void main(String[] args) {
TextThread3 thread3 = new TextThread3();
}
}

與第2種方式比較,我們發現Frame類不再實現Runnable接口了,而是在實例化Thread類的時候,定義了一個實現Runnable接口的匿名內部類:
復制代碼 代碼如下:

clockThread = new Thread(new Runnable() {
@Override
public void run() {
while (isRunning) {
try {
Thread.currentThread().sleep(1000);
timer ++;
label.setText("逝去了:"+timer);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
});

有關Java多線程的內容還有很多,例如線程優先級、線程同步等,由于這些內容與本書關系不是很緊密,所以不再介紹了,有關其他的線程知識可以參考Java方面的書籍。接下來介紹一下Android中的線程。
Android中的線程
  在Android平臺中多線程應用很廣泛,在UI更新、游戲開發和耗時處理(網絡通信等)等方面都需要多線程。Android線程涉及的技術有:Handler;Message;MessageQueue;Looper;HandlerThread。
  Android線程應用中的問題與分析
  為了介紹這些概念,我們把計時器的案例移植到Android系統上,按照在Frame方式修改之后的代碼清單.
復制代碼 代碼如下:

package com.example.testthread4;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.*;
import android.app.Activity;
import android.view.Menu;
import android.view.View;
import android.view.View.OnClickListener;
public class MainActivity extends Activity {
private Button mButton;
private TextView mTextView;
private boolean isRunning = true;
private Thread mThread;
private int timer = 0;
//private Handler handler;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mButton = (Button) findViewById(R.id.button1);
mTextView = (TextView) findViewById(R.id.textView1);
mButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View arg0) {
// TODO Auto-generated method stub
isRunning = false;
}
});
/*handler = new Handler() {
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case 0:
mTextView.setText("逝去了:" + msg.obj);
}
}
};*/
mThread = new Thread(new Runnable() {
@Override
public void run() {
try {
while (isRunning) {
Thread.currentThread().sleep(1000);
timer++;
/*Message message = new Message();
message.obj = timer;
message.what = 0;
handler.sendMessage(message);*/
mTextView.setText("逝去了:"+timer);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
mThread.start();
}
}

程序打包運行結果出現了異常,如圖所示。
 
們打開LogCat窗口,出錯日志信息如圖所示。
 
系統拋出的異常信息是“Only the original thread that created a view hierarchy can touch its views”,在Android中更新UI處理必須由創建它的線程更新,而不能在其他線程中更新。上面的錯誤原因就在于此。
  現在分析一下上面的案例,在上面的程序中有兩個線程:一個主線程和一個子線程,它們的職責如圖所示。
  由于labelTimer是一個UI控件,它是在主線程中創建的,但是它卻在子線程中被更新了,更新操作在clockThread線程的run()方法中實現,代碼如下:
 
復制代碼 代碼如下:

mThread = new Thread(new Runnable() {
@Override
public void run() {
try {
while (isRunning) {
Thread.currentThread().sleep(1000);
timer++;
/*Message message = new Message();
message.obj = timer;
message.what = 0;
handler.sendMessage(message);*/
mTextView.setText("逝去了:"+timer);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});

這樣的處理違背了Android多線程編程規則,系統會拋出異常“Only the original thread that created a view hierarchy can touch its views”。
  要解決這個問題,就要明確主線程和子線程的職責。主線程的職責是創建、顯示和更新UI控件、處理UI事件、啟動子線程、停止子線程;子線程的職責是計算逝去的時間和向主線程發出更新UI消息,而不是直接更新UI。
主線程的職責是顯示UI控件、處理UI事件、啟動子線程、停止子線程和更新UI,子線程的職責是計算逝去的時間和向主線程發出更新UI消息。但是新的問題又出現了:子線程和主線程如何發送消息、如何通信呢?
在Android中,線程有兩個對象―消息(Message)和消息隊列(MessageQueue)可以實現線程間的通信。下面再看看修改之后的代碼清單.
復制代碼 代碼如下:

package com.example.testthread4;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.*;
import android.app.Activity;
import android.view.Menu;
import android.view.View;
import android.view.View.OnClickListener;
public class MainActivity extends Activity {
private Button mButton;
private TextView mTextView;
private boolean isRunning = true;
private Thread mThread;
private int timer = 0;
private Handler handler;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mButton = (Button) findViewById(R.id.button1);
mTextView = (TextView) findViewById(R.id.textView1);
mButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View arg0) {
// TODO Auto-generated method stub
isRunning = false;
}
});
handler = new Handler() {
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case 0:
mTextView.setText("逝去了:" + msg.obj);
}
}
};
mThread = new Thread(new Runnable() {
@Override
public void run() {
try {
while (isRunning) {
Thread.currentThread().sleep(1000);
timer++;
Message message = new Message();
message.obj = timer;
message.what = 0;
handler.sendMessage(message);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
mThread.start();
}
}

有的時候為了將Android代碼變得更加緊湊,把線程的創建和啟動編寫在一條語句中,如下面
復制代碼 代碼如下:

/*mThread = */new Thread(new Runnable() {
@Override
public void run() {
try {
while (isRunning) {
Thread.currentThread().sleep(1000);
timer++;
Message message = new Message();
message.obj = timer;
message.what = 0;
handler.sendMessage(message);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}).start();

運行模擬器結果如圖8-1所示,加載屏幕后馬上開始計時,也可以單擊“停止計時”按鈕來停止計時。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美日韩一区二区免费在线观看| 97精品国产97久久久久久免费| 久久av资源网站| 日韩av一区在线| 亚洲欧美综合另类中字| 精品小视频在线| 亚洲国产精品女人久久久| 国产脚交av在线一区二区| 久久精品国产69国产精品亚洲| 日韩免费观看高清| 国产精品永久免费在线| 国产精品免费看久久久香蕉| 中文字幕在线日韩| 欧美一区二区大胆人体摄影专业网站| 国产a级全部精品| 国产精品91久久久久久| 日韩毛片在线观看| 欧美激情视频网站| 亚洲人成在线观| 日韩中文有码在线视频| 中文字幕成人在线| 92福利视频午夜1000合集在线观看| 亚洲精品mp4| 亚洲欧美日本精品| 欧美综合第一页| 日韩人在线观看| 日本久久久久亚洲中字幕| 精品国产依人香蕉在线精品| 国产一区二区三区在线观看网站| 欧美黑人国产人伦爽爽爽| 成人网页在线免费观看| 久久av中文字幕| 欧美乱大交xxxxx| 日本中文字幕成人| 国产精品av网站| 亚洲一区中文字幕| 色婷婷久久一区二区| 欧美性少妇18aaaa视频| 国产精品视频公开费视频| 欧美国产一区二区三区| 日韩一中文字幕| 国内揄拍国内精品| 久久精品在线播放| 日韩欧美a级成人黄色| 欧美日韩美女在线观看| 精品亚洲一区二区三区在线观看| 亚洲黄色在线观看| 久久夜精品香蕉| 国产做受69高潮| 亚州精品天堂中文字幕| 亚洲网址你懂得| 欧美激情精品久久久久久大尺度| 欧美刺激性大交免费视频| 久久久久五月天| 国产精品视频久久久| 日韩欧美在线观看| 欧美专区在线视频| 久久影视电视剧凤归四时歌| 欧美激情2020午夜免费观看| 久久免费视频这里只有精品| 国精产品一区一区三区有限在线| 91欧美视频网站| 日韩成人av在线| 欧美日韩中文字幕日韩欧美| 国产成人91久久精品| 久久久久久这里只有精品| 亚洲美女av电影| 欧美综合在线第二页| 91国内免费在线视频| 欧美黑人巨大xxx极品| 欧美国产在线电影| 亚洲国产精品va在线看黑人动漫| 欧美精品中文字幕一区| 亚洲人永久免费| 91精品久久久久久久久久另类| 国内精品国产三级国产在线专| 国语自产精品视频在线看| 在线观看免费高清视频97| 亚洲第一中文字幕| 少妇激情综合网| 色青青草原桃花久久综合| 欧美精品一区二区三区国产精品| 亚洲精品中文字幕有码专区| 欧美激情网友自拍| 亚洲国产古装精品网站| 黑人极品videos精品欧美裸| 国产999在线| 欧美一级大片视频| 欧美日韩一区二区免费在线观看| 欧美黑人国产人伦爽爽爽| 亚洲夜晚福利在线观看| 国产亚洲免费的视频看| 久热精品视频在线免费观看| 国产69精品久久久久9| 久久九九全国免费精品观看| 欧美视频中文字幕在线| 欧美日本黄视频| 亚洲国产成人91精品| 美日韩丰满少妇在线观看| 欧美日韩国产综合新一区| 亚洲欧美激情另类校园| 精品一区二区三区四区在线| 久久视频国产精品免费视频在线| 最近中文字幕mv在线一区二区三区四区| 亚洲成人1234| 超在线视频97| 热99久久精品| 国产精品爽爽爽爽爽爽在线观看| 亚洲第一天堂无码专区| 日韩av在线高清| 国产亚洲精品美女久久久久| 欧美性一区二区三区| 中文字幕精品视频| 欧美性xxxx极品hd满灌| 日韩大陆欧美高清视频区| 久久在线免费观看视频| 国产成人avxxxxx在线看| 精品国模在线视频| 精品亚洲男同gayvideo网站| 国产日韩av高清| 国产成人avxxxxx在线看| 成人av.网址在线网站| 这里只有精品久久| 久久久这里只有精品视频| 久久人人爽人人爽人人片av高清| 91黄色8090| 欧美成年人视频网站欧美| 色噜噜狠狠狠综合曰曰曰88av| 精品视频在线播放| 欧美性生交大片免费| 亚洲精品国产电影| 精品国产乱码久久久久久虫虫漫画| 91午夜在线播放| 国产69久久精品成人看| 国产成人精品免费久久久久| 久久九九全国免费精品观看| 欧美福利小视频| 九九热视频这里只有精品| 国产精品视频免费在线| 久久久精品视频在线观看| 成人国产亚洲精品a区天堂华泰| 国产婷婷97碰碰久久人人蜜臀| 欧美美女18p| 成人在线免费观看视视频| 欧美日本精品在线| 欧美日韩高清在线观看| 91av在线播放视频| 亚洲人成在线一二| 成人啪啪免费看| 国产精品高清在线| 久久免费少妇高潮久久精品99| 69久久夜色精品国产69乱青草| 成人激情在线观看| 中文字幕欧美日韩va免费视频| 亚洲人成在线一二| 国产精品久久久久久久午夜| 岛国视频午夜一区免费在线观看| 久久精品国产欧美激情| 国产婷婷成人久久av免费高清| 国产精品白丝jk喷水视频一区| 欧美激情精品久久久久久黑人| 精品日本美女福利在线观看| 国产午夜精品视频| 久久久精品影院|