青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

C++ Coder

HCP高性能計算架構,實現,編譯器指令優化,算法優化, LLVM CLANG OpenCL CUDA OpenACC C++AMP OpenMP MPI

C++博客 首頁 新隨筆 聯系 聚合 管理
  98 Posts :: 0 Stories :: 0 Comments :: 0 Trackbacks
http://www.cnblogs.com/keyindex/articles/1822463.html

android的Handler

前言

  學習android一段時間了,為了進一步了解android的應用是如何設計開發的,決定詳細研究幾個開源的android應用。從一些開源應用中吸收點東西,一邊進行量的積累,一邊探索android的學習研究方向。這里我首先選擇了jwood的 Standup Timer 項目。本文將把研究的內容筆記整理,建立一個索引列表。

關鍵詞

  Android.os.Handler涉及較多的知識點,我把一些關鍵詞列舉在下面,將主要介紹Handler:

android.os.Handler

  Handler在android里負責發送和處理消息。它的主要用途有:
  1)按計劃發送消息或執行某個Runnanble(使用POST方法);
  2)從其他線程中發送來的消息放入消息隊列中,避免線程沖突(常見于更新UI線程)
   默認情況下,Handler接受的是當前線程下的消息循環實例(使用Handler(Looper looper)、Handler(Looper looper, Handler.Callback callback)可以指定線程),同時一個消息隊列可以被當前線程中的多個對象進行分發、處理(在UI線程中,系統已經有一個Activity來處理了,你可以再起若干個Handler來處理)。在實例化Handler的時候,Looper可以是任意線程的,只要有Handler的指針,任何線程也都可以sendMessage。Handler對于Message的處理不是并發的。一個Looper 只有處理完一條Message才會讀取下一條,所以消息的處理是阻塞形式的(handleMessage()方法里不應該有耗時操作,可以將耗時操作放在其他線程執行,操作完后發送Message(通過sendMessges方法),然后由handleMessage()更新UI)。

倒計時程序

  利用Timer 編寫一個倒計時程序,程序使用Timer和TimerTask來完成倒計時,同時使用sendMessages方法發送消息,然后在HanleMessage里更新UI。
Activity布局:
Layout
復制代碼
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation
="vertical"
android:layout_width
="fill_parent"
android:layout_height
="fill_parent"
>
<TextView 
android:layout_width="fill_parent" 
android:layout_height
="wrap_content" 
android:layout_gravity
="center"
android:id
="@+id/txt"
/>
<Button
android:id="@+id/btnStartTime"
android:text
="開始計時"
android:layout_width
="80dip"
android:layout_height
="wrap_content" 

></Button>
<Button
android:id="@+id/btnStopTime"
android:text
="停止計時"
android:layout_width
="80dip"
android:layout_height
="wrap_content"
/>

<SeekBar android:id="@+id/SeekBar01" android:layout_width="match_parent" android:layout_height="wrap_content"></SeekBar>
</LinearLayout>
復制代碼

 

這里使用TextView 來顯示倒計時的時間變化,兩個按鈕用于控制時間的開始和停止。SeekBar主要是用于查看線程是否被阻塞(阻塞時無法拖動)。
 
onCreate
復制代碼
@Override
publicvoid onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
txt 
= (TextView) findViewById(R.id.txt);
btnStart 
= (Button) findViewById(R.id.btnStartTime);
btnStop 
= (Button) findViewById(R.id.btnStopTime);
Log.d(
"ThreadId""onCread:"
+ String.valueOf(Thread.currentThread().getId()));
myHandler 
=new Handler(this);

btnStart.setOnClickListener(
this);
btnStop.setOnClickListener(
this);

}
復制代碼

 

在onCreate方法中初始化元素個元素,myHandler = new Handler(this); 調用的是  Handler(Handler.Callback callback)構造函數,在回調方法callback中對發送來的消息進行處理(這樣我們就不必使用內部類的寫法來 重寫HandleMessage()方法了),因此Activity必須實現 android.os.Handler.Callback 接口。我們還在將onCreate 方法的ThreadId 記錄在了Log中用以和消息發送、處理時所作的線程進行比較。
 
發送消息
復制代碼

@Override
publicvoid onClick(View v) {
switch (v.getId()) {
case R.id.btnStartTime:
startTimer();
break;
case R.id.btnStopTime:
timer.cancel();

break;
}

}

privatesynchronizedvoid startTimer() {

timer 
=new Timer();
// TimerTask updateTimerValuesTask = new TimerTask() {
// @Override
// public void run() {
// updateTimerValues();
// }
//
// };
//自定義的CallBack模式。Task繼承自TimerTask
Task updateTimerValuesTask =new Task(this);

timer.schedule(updateTimerValuesTask, 
10001000);
}

//執行耗時的倒計時任務。
privatevoid updateTimerValues() {
total
--;

Log.d(
"ThreadId""send:"
+ String.valueOf(Thread.currentThread().getId()));

Message msg
=new Message();
Bundle date 
=new Bundle();// 存放數據
date.putInt("time", total);
msg.setData(date);
msg.what
=0;
myHandler.sendMessage(msg);

//另一種寫法
// Message msg=myHandler.obtainMessage();
// Bundle date = new Bundle();// 存放數據
// date.putInt("time", total);
// msg.setData(date);
// msg.what=0;
// msg.sendToTarget();

}

@Override
publicvoid TaskRun() {
updateTimerValues();

}

復制代碼

 

實現Button按鈕的事件處理以此進入倒計時操作。這里使用的Timer 來執行定時操作(其實我們完全可以另起一個線程)。Task類繼承了TimerTask類,里面增加了一個任務處理接口來實現回調模式,應此Activity需要實現該回調的接口 ITaskCallBack(這樣做是因為我比較不喜歡內部類的編寫方法)。
ICallBack接口和Task類
復制代碼
publicinterface ITaskCallBack {

void TaskRun();
}



publicclass Task extends TimerTask {

private ITaskCallBack iTask;

public Task(ITaskCallBack iTaskCallBack)
{
super();
iTask
=iTaskCallBack;
}

publicvoid setCallBack(ITaskCallBack iTaskCallBack)
{
iTask
=iTaskCallBack;
}
@Override
publicvoid run() {
// TODO Auto-generated method stub
iTask.TaskRun();
}

}
復制代碼

 

這是Java的回調函數的一般寫法。
 
實現CallBack
復制代碼

/**
* 實現消息處理
*/
@Override
publicboolean handleMessage(Message msg) {

switch(msg.what)
{
case0:
Bundle date
=msg.getData();
txt.setText(String.valueOf(date.getInt(
"time")));

Log.d(
"ThreadId""HandlerMessage:"
+ String.valueOf(Thread.currentThread().getId()));
Log.d(
"ThreadId""msgDate:"
+ String.valueOf(date.getInt("time")));
break;

}
returnfalse;
}
復制代碼

 

  可以看到 實現 android.os.Handler.Callback 接口,其實就是對handleMessage()方法進行重寫(和內部類的一個區別是,內部類的返回值是Void)。

運行結果

  可以看到在onCreate 方法中線程的ID是1(UI線程) 這與 HandlerMessage 進行消息處理時是所作的線程ID是一樣的,而消息發送的線程ID則為8非UI線程。

使用Threadle進行實現

Activity類
復制代碼

publicclass ThreadHandlerrActivity extends Activity implements Callback,
OnClickListener {

private TextView txt;
private Button btnStart, btnStop;
private Handler myHandler;
private TimerThread timerThread;
privateint Total=30;


/** Called when the activity is first created. */
@Override
publicvoid onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
txt 
= (TextView) findViewById(R.id.txt);
btnStart 
= (Button) findViewById(R.id.btnStartTime);
btnStop 
= (Button) findViewById(R.id.btnStopTime);
Log.d(
"ThreadId""onCread:"
+ String.valueOf(Thread.currentThread().getId()));
myHandler 
=new Handler(this);


btnStart.setOnClickListener(
this);
btnStop.setOnClickListener(
this);

}

/**
* 實現消息處理
*/
@Override
publicboolean handleMessage(Message msg) {

switch(msg.what)
{
case0:
Bundle date
=msg.getData();
txt.setText(String.valueOf(date.getInt(
"time")));

Log.d(
"ThreadId""HandlerMessage:"
+ String.valueOf(Thread.currentThread().getId()));
Log.d(
"ThreadId""msgDate:"
+ String.valueOf(date.getInt("time")));
break;

}
returnfalse;
}

@Override
publicvoid onClick(View v) {
switch (v.getId()) {
case R.id.btnStartTime:
//自定義的線程
timerThread=new TimerThread(myHandler,60);
timerThread.start();

break;
case R.id.btnStopTime:
timerThread.stop();
//timerThread.destroy();
break;
}

}




}
復制代碼

 

自定義的線程類
復制代碼
**
* 自定義的線程類,通過傳入的Handler,和Total 定期執行耗時操作
* @author linzijun
*
*/
publicclass TimerThread extends Thread {

publicint Total=60;
public Handler handler;
/**
* 初始化構造函數
@param mhandler handler 用于發送消息
@param total 總周期
*/
public TimerThread(Handler mhandler,int total)
{
super();
handler
=mhandler;
Total
=total;
}
@Override
publicvoid run() {

while(true)
{
Total
--;
if(Total<0)
break;
try {
Thread.sleep(
1000);
catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Message msg
=new Message();
Bundle date 
=new Bundle();// 存放數據
date.putInt("time", Total);
msg.setData(date);
msg.what
=0;
Log.d(
"ThreadId""Thread:"
+ String.valueOf(Thread.currentThread().getId()));
handler.sendMessage(msg);


}

super.run();
}



}
復制代碼

 

這里繼承了Thread類,也可以直接實現 Runnable接口。

關于POST

  Post的各種方法是把一個Runnable發送給消息隊列,它將在到達時進行處理。
POST
復制代碼

publicclass PostHandler extends Activity implements OnClickListener, Runnable {

private TextView txt;
private Button btnStart, btnStop;
private Handler myHandler;
private Timer timer;
privateint total =60;


@Override
protectedvoid onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);

setContentView(R.layout.main);
txt 
= (TextView) findViewById(R.id.txt);
btnStart 
= (Button) findViewById(R.id.btnStartTime);
btnStop 
= (Button) findViewById(R.id.btnStopTime);
Log.d(
"ThreadId""onCread:"
+ String.valueOf(Thread.currentThread().getId()));
myHandler 
=new Handler()
{

@Override
publicvoid handleMessage(Message msg) {
switch(msg.what)
{
case0:
Bundle date
=msg.getData();
txt.setText(String.valueOf(date.getInt(
"time")));

Log.d(
"ThreadId""HandlerMessage:"
+ String.valueOf(Thread.currentThread().getId()));
Log.d(
"ThreadId""msgDate:"
+ String.valueOf(date.getInt("time")));
break;

}

}

};

btnStart.setOnClickListener(
this);
btnStop.setOnClickListener(
this);
}

@Override
publicvoid onClick(View v) {
switch (v.getId()) {
case R.id.btnStartTime:
//myHandler.post(this);
myHandler.postDelayed(this1000);
break;
case R.id.btnStopTime:

break;
}

}

@Override
publicvoid run() {
while(true)
{
total
--;
if(total<0)
break;
try {
Thread.sleep(
1000);
catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Message msg
=new Message();
Bundle date 
=new Bundle();// 存放數據
date.putInt("time", total);
msg.setData(date);
msg.what
=0;
Log.d(
"ThreadId""POST:"
+ String.valueOf(Thread.currentThread().getId()));
myHandler.sendMessage(msg);
Log.d(
"ThreadId""Thread:"
+ String.valueOf(Thread.currentThread().getId()));

}

}

}
復制代碼

 

使用POST的方式 是將Runnable 一起發送給處理的線程(這里為UI),如果Runnable的操作比較耗時的話那線程將進入阻塞狀態??梢钥吹较冗\行 Runnable的Run方法 然后在進入 HandleMessage() 。我還嘗試了另一種寫法,將TimerThreadPOST過去,運行結果是一樣的。
代碼
復制代碼
package zijunlin.me;

import java.util.Timer;

import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;

publicclass PostHandler extends Activity implements OnClickListener, Runnable {

private TextView txt;
private Button btnStart, btnStop;
private Handler myHandler;
private Timer timer;
privateint total =60;
private TimerThread timerThread;

@Override
protectedvoid onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);

setContentView(R.layout.main);
txt 
= (TextView) findViewById(R.id.txt);
btnStart 
= (Button) findViewById(R.id.btnStartTime);
btnStop 
= (Button) findViewById(R.id.btnStopTime);
Log.d(
"ThreadId""onCread:"
+ String.valueOf(Thread.currentThread().getId()));
myHandler 
=new Handler()
{

@Override
publicvoid handleMessage(Message msg) {
switch(msg.what)
{
case0:
Bundle date
=msg.getData();
txt.setText(String.valueOf(date.getInt(
"time")));

Log.d(
"ThreadId""HandlerMessage:"
+ String.valueOf(Thread.currentThread().getId()));
Log.d(
"ThreadId""msgDate:"
+ String.valueOf(date.getInt("time")));
break;

}

}

};

btnStart.setOnClickListener(
this);
btnStop.setOnClickListener(
this);
}

@Override
publicvoid onClick(View v) {
switch (v.getId()) {
case R.id.btnStartTime:
//myHandler.post(this);
//myHandler.postDelayed(this, 1000);
timerThread=new TimerThread(myHandler,60);

myHandler.post(timerThread);
break;
case R.id.btnStopTime:

break;
}

}

@Override
publicvoid run() {
while(true)
{
total
--;
if(total<0)
break;
try {
Thread.sleep(
1000);
catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Message msg
=new Message();
Bundle date 
=new Bundle();// 存放數據
date.putInt("time", total);
msg.setData(date);
msg.what
=0;
Log.d(
"ThreadId""POST:"
+ String.valueOf(Thread.currentThread().getId()));
myHandler.sendMessage(msg);
Log.d(
"ThreadId""Thread:"
+ String.valueOf(Thread.currentThread().getId()));

}

}

}
復制代碼

 

可以說POST的各種方法主要是用于 “按計劃發送消息或執行某個Runnanble(使用POST方法)”。

參考文獻

  SDK
posted on 2013-04-24 10:55 jackdong 閱讀(757) 評論(0)  編輯 收藏 引用 所屬分類: Android

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲美女av黄| 欧美在线一二三| 性色一区二区| 午夜精品美女自拍福到在线 | 99国产精品| 亚洲狼人精品一区二区三区| 亚洲精品人人| 亚洲一品av免费观看| 午夜精品一区二区三区在线播放 | 亚洲国产成人tv| 久久免费视频一区| 欧美国产欧美亚洲国产日韩mv天天看完整 | 亚洲成人在线免费| 亚洲日本中文字幕免费在线不卡| 99xxxx成人网| 午夜精品一区二区三区在线| 欧美在线中文字幕| 欧美 日韩 国产 一区| 欧美日韩一本到| 国产综合香蕉五月婷在线| 亚洲人午夜精品| 欧美一区二区三区精品| 欧美激情精品久久久久久蜜臀| 宅男噜噜噜66一区二区| 麻豆av一区二区三区| 国产精品欧美日韩| 亚洲毛片在线观看.| 久久永久免费| 亚洲一区二区毛片| 欧美精品在线一区二区| 国产综合自拍| 亚洲欧美日韩中文播放| 亚洲成人在线视频播放 | 亚洲午夜激情在线| 久久久久国产精品麻豆ai换脸| 欧美日韩视频在线| 伊人久久亚洲美女图片| 亚洲一区成人| 欧美精品电影| 老司机成人在线视频| 欧美久久九九| 黄色亚洲在线| 久久av在线| 一区二区三欧美| 欧美激情亚洲综合一区| 亚洲福利电影| 久久久久久噜噜噜久久久精品 | 一区二区激情视频| 欧美成人精品一区二区| 娇妻被交换粗又大又硬视频欧美| 在线一区二区三区四区| 亚洲国产精彩中文乱码av在线播放| 午夜激情综合网| 国产精品成人v| 一区二区三区日韩精品视频| 欧美黄色aaaa| 另类专区欧美制服同性| 在线成人黄色| 麻豆国产精品va在线观看不卡| 亚洲欧美在线磁力| 国产欧美日韩中文字幕在线| 亚洲欧美日韩精品综合在线观看 | 久久久999精品免费| 99视频热这里只有精品免费| 欧美激情在线有限公司| 亚洲美女av网站| 亚洲片在线观看| 欧美精品亚洲精品| 一区二区三区你懂的| 日韩一级片网址| 国产精品大片wwwwww| 午夜视频久久久| 久久精品二区三区| 亚洲黄色影院| 日韩一级片网址| 国产精品婷婷| 久久久久综合网| 老鸭窝毛片一区二区三区| 日韩视频中午一区| 一区二区三区欧美激情| 国产性色一区二区| 亚洲二区三区四区| 国产精品久久久久9999吃药| 久久不射中文字幕| 免费观看成人| 亚洲男同1069视频| 欧美专区第一页| 日韩天堂在线视频| 午夜视频久久久久久| 最新日韩av| 亚洲一区二区三区久久| 在线观看亚洲视频啊啊啊啊| 日韩午夜三级在线| 国产永久精品大片wwwapp| 亚洲国产精品成人精品| 国产精品永久| 亚洲国产你懂的| 欧美日本不卡| 久久精品国产精品| 女人香蕉久久**毛片精品| 一区二区国产日产| 欧美一区=区| 一本色道久久综合| 久久久久久久综合色一本| 这里只有精品视频在线| 久久久久国产精品麻豆ai换脸| 中文国产亚洲喷潮| 久久久无码精品亚洲日韩按摩| 一区二区三区四区精品| 久久天天躁狠狠躁夜夜爽蜜月| 亚洲欧美日本视频在线观看| 欧美高清在线一区| 久久久精品一品道一区| 欧美性一区二区| 亚洲精品黄色| 亚洲日韩中文字幕在线播放| 久久精品国产亚洲一区二区| 香蕉久久夜色精品国产使用方法| 欧美另类专区| 欧美激情一二三区| 国精品一区二区三区| 亚洲欧美日本在线| 亚洲一区二区影院| 欧美激情在线免费观看| 亚洲国产成人久久综合一区| 一区二区三区在线观看视频| 午夜久久黄色| 久久国产色av| 国产精品婷婷午夜在线观看| 99在线观看免费视频精品观看| 亚洲精品中文字幕在线| 欧美freesex8一10精品| 欧美成人免费大片| 亚洲第一页中文字幕| 久久在线视频在线| 欧美不卡在线视频| 1024国产精品| 久久综合亚洲社区| 欧美成年人视频| 亚洲国产精品嫩草影院| 看片网站欧美日韩| 欧美激情按摩| 亚洲人成在线播放| 欧美国产视频一区二区| 亚洲国产天堂网精品网站| 91久久在线播放| 蜜臀91精品一区二区三区| 欧美国产先锋| 一本色道久久综合亚洲精品小说| 欧美日韩精品免费看| 亚洲午夜久久久久久尤物 | 亚洲国产精品国自产拍av秋霞| 久久久美女艺术照精彩视频福利播放 | 亚洲视频久久| 欧美一区二区视频在线| 国产综合色一区二区三区| 蜜臀av性久久久久蜜臀aⅴ四虎| 亚洲国产影院| 欧美一区二区三区在线免费观看| 亚洲高清av| 欧美第一黄色网| 亚洲另类自拍| 欧美在线看片a免费观看| 国产乱码精品一区二区三区不卡| 欧美一级精品大片| 亚洲第一黄网| 午夜精品福利在线观看| 国产在线精品一区二区中文 | 久久综合久久综合久久综合| 亚洲国产99精品国自产| 亚洲婷婷综合久久一本伊一区| 国产精品久久久久久久久久久久久久| 亚洲小说春色综合另类电影| 美女福利精品视频| 亚洲一本大道在线| 亚洲高清视频的网址| 国产精品福利在线| 久久人人爽爽爽人久久久| 99国产精品国产精品久久| 欧美在线播放高清精品| 亚洲国产日韩精品| 国产精品自拍小视频| 蜜桃av久久久亚洲精品| 亚洲在线观看免费视频| 亚洲国产激情| 裸体一区二区三区| 亚洲一级特黄| 亚洲人成啪啪网站| 激情欧美日韩| 国产欧美日韩在线| 国产精品女主播| 欧美激情1区| 久久久综合香蕉尹人综合网| 亚洲欧美日韩综合一区| 99这里只有久久精品视频| 欧美国产激情二区三区| 久久精品国产99| 亚洲欧美国产三级| 99在线观看免费视频精品观看| 亚洲大胆视频|