Android中Handler的使用

本文详细解释了Android OS中Handler的作用及其在消息队列和线程队列中的应用,通过一个进度条Demo展示了如何使用Handler开启线程并在主线程中更新UI。程序运行结果验证了Handler在同一线程中的执行情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Handler 为Android操作系统中的线程通信工具,包为android.os.Handler。

与Handler绑定的有两个队列,一个为消息队列,另一个为线程队列。Handler可以通过这两个队列来分别:

  1. 发送、接受、处理消息–消息队列;
  2. 启动、结束、休眠线程–线程队列;

Android OS中,一个进程被创建之后,主线程(可理解为当前Activity)创建一个消息队列,这个消息队列维护所有顶层应用对象(Activities, Broadcast receivers等)以及主线程创建的窗口。你可以在主线程中创建新的线程,这些新的线程都通过Handler与主线程进行通信。通信通过新线程调用 Handler的post()方法和sendMessage()方法实现,分别对应功能:

  1. post()  将一个线程加入线程队列;
  2. sendMessage() 发送一个消息对象到消息队列;

当然,post()方法还有一些变体,比如postDelayed()、postAtTime()分别用来延迟发送、定时发送;

消息的处理,在主线程的Handler对象中进行;具体处理过程,需要在new Handler对象时使用匿名内部类重写Handler的handleMessage(Message msg)方法;

线程加入线程队列可以在主线程中也可以在子线程中进行,但都要通过主线程的Handler对象调用post()。

下面我以一个进度条Demo来展示Handler的使用,每隔2000ms就使进度条进一格,先预览下程序结构图:

[1] main.xml中的布局文件源码如下:

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:orientation="vertical"  
  4.     android:layout_width="fill_parent"  
  5.     android:layout_height="fill_parent"  
  6.     >  
  7. <TextView    
  8.     android:layout_width="fill_parent"   
  9.     android:layout_height="wrap_content"   
  10.     android:textSize="16sp"  
  11.     android:text="Hello , This is Andy's blog !"/>  
  12.     <Button  
  13.        android:id="@+id/start"  
  14.        android:layout_width="fill_parent"  
  15.        android:layout_height="wrap_content"  
  16.        android:text="Start"/>  
  17.     <Button  
  18.        android:id="@+id/end"  
  19.        android:layout_width="fill_parent"  
  20.        android:layout_height="wrap_content"  
  21.        android:text="End"/>    
  22.     <ProgressBar  
  23.        android:id="@+id/pBar"  
  24.        android:layout_width="fill_parent"  
  25.        android:layout_height="wrap_content"  
  26.        style="?android:attr/progressBarStyleHorizontal"  
  27.        mce_style="?android:attr/progressBarStyleHorizontal"  
  28.        android:visibility="gone"/>  
  29. </LinearLayout>  

<?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:textSize="16sp" android:text="Hello , This is Andy's blog !"/> <Button android:id="@+id/start" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="Start"/> <Button android:id="@+id/end" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="End"/> <ProgressBar android:id="@+id/pBar" android:layout_width="fill_parent" android:layout_height="wrap_content" style="?android:attr/progressBarStyleHorizontal" mce_style="?android:attr/progressBarStyleHorizontal" android:visibility="gone"/> </LinearLayout>

 

[2] HandlerActivity.java中的源码如下:

  1. package com.andyidea.handlerdemo;  
  2.   
  3. import android.app.Activity;  
  4. import android.os.Bundle;  
  5. import android.os.Handler;  
  6. import android.os.Message;  
  7. import android.util.Log;  
  8. import android.view.View;  
  9. import android.widget.Button;  
  10. import android.widget.ProgressBar;  
  11.   
  12. public class HandlerActivity extends Activity {  
  13.       
  14.     Button btnStart,btnEnd;  
  15.     ProgressBar proBar;  
  16.       
  17.     /** Called when the activity is first created. */  
  18.     @Override  
  19.     public void onCreate(Bundle savedInstanceState) {  
  20.         super.onCreate(savedInstanceState);  
  21.         setContentView(R.layout.main);  
  22.           
  23.         //通过控件的ID来实例化控件对象  
  24.          btnStart = (Button)findViewById(R.id.start);  
  25.         btnEnd = (Button)findViewById(R.id.end);  
  26.         proBar = (ProgressBar)findViewById(R.id.pBar);  
  27.           
  28.         //开始按钮触发事件  
  29.         btnStart.setOnClickListener(new View.OnClickListener() {  
  30.               
  31.             @Override  
  32.             public void onClick(View v) {  
  33.                 proBar.setVisibility(View.VISIBLE);  
  34.                 updateBarHandler.post(updateBarThread);  
  35.             }  
  36.         });  
  37.           
  38.         //结束按钮触发事件  
  39.         btnEnd.setOnClickListener(new View.OnClickListener() {  
  40.               
  41.             @Override  
  42.             public void onClick(View v) {  
  43.                 updateBarHandler.removeCallbacks(updateBarThread);  
  44.             }  
  45.         });  
  46.     }  
  47.       
  48.       
  49.      //创建一个Handler对象  
  50.      Handler updateBarHandler = new Handler(){  
  51.   
  52.         @Override  
  53.         public void handleMessage(Message msg) {  
  54.             proBar.setProgress(msg.arg1);  
  55.             updateBarHandler.post(updateBarThread);  
  56.         }  
  57.           
  58.     };  
  59.       
  60.     //更新ProgressBar的线程对象  
  61.     Runnable updateBarThread = new Runnable() {  
  62.         int i = 0;  
  63.         @Override  
  64.         public void run() {  
  65.             i = i + 10;  
  66.             Message msg = updateBarHandler.obtainMessage();  
  67.             msg.arg1 = i;  
  68.             try{  
  69.                 Thread.sleep(2000);  
  70.             }catch (InterruptedException e) {  
  71.                 e.printStackTrace();  
  72.             }  
  73.             updateBarHandler.sendMessage(msg);  
  74.             if(i == 100){  
  75.                 updateBarHandler.removeCallbacks(updateBarThread);  
  76.             }  
  77.         }  
  78.     };  
  79. }  

package com.andyidea.handlerdemo; 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.widget.Button; import android.widget.ProgressBar; public class HandlerActivity extends Activity { Button btnStart,btnEnd; ProgressBar proBar; /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); //通过控件的ID来实例化控件对象 btnStart = (Button)findViewById(R.id.start); btnEnd = (Button)findViewById(R.id.end); proBar = (ProgressBar)findViewById(R.id.pBar); //开始按钮触发事件 btnStart.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { proBar.setVisibility(View.VISIBLE); updateBarHandler.post(updateBarThread); } }); //结束按钮触发事件 btnEnd.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { updateBarHandler.removeCallbacks(updateBarThread); } }); } //创建一个Handler对象 Handler updateBarHandler = new Handler(){ @Override public void handleMessage(Message msg) { proBar.setProgress(msg.arg1); updateBarHandler.post(updateBarThread); } }; //更新ProgressBar的线程对象 Runnable updateBarThread = new Runnable() { int i = 0; @Override public void run() { i = i + 10; Message msg = updateBarHandler.obtainMessage(); msg.arg1 = i; try{ Thread.sleep(2000); }catch (InterruptedException e) { e.printStackTrace(); } updateBarHandler.sendMessage(msg); if(i == 100){ updateBarHandler.removeCallbacks(updateBarThread); } } }; }

程序运行的截图如下:

注:在主线程HandlerActivity中,通过Handler对象将updateBarThread子线程对象添加到主线程的队列中,并不是另外开启了一个新的线程去执行。

 本文通过例子来验证和说明使用Handler对象开启的线程是否和主线程为在同一线程中。

程序结构图如下:

[1] HandlerActivity.java中的源码如下:

  1. package com.andyidea.handlerdemo2;  
  2.   
  3. import android.app.Activity;  
  4. import android.os.Bundle;  
  5. import android.os.Handler;  
  6. import android.os.HandlerThread;  
  7. import android.os.Looper;  
  8. import android.os.Message;  
  9. import android.util.Log;  
  10.   
  11. public class HandlerActivity extends Activity {  
  12.       
  13.     private Handler handler = new Handler();  
  14.       
  15.     @Override  
  16.     public void onCreate(Bundle savedInstanceState) {  
  17.         super.onCreate(savedInstanceState);  
  18.         //handler.post(r);  
  19.         setContentView(R.layout.main);  
  20.         Log.e("activity-->" ,Thread.currentThread().getId() + "");  
  21.         Log.e("activityname-->" ,Thread.currentThread().getName() + "");  
  22.          
  23.         //与主线程同一线程  
  24.         handler.post(r);  
  25.           
  26.         //与主线程不同线程  
  27.         HandlerThread handlerThread = new HandlerThread("handler_thread");  
  28.         handlerThread.start();  
  29.         //HandlerThread线程对象要启动后,  
  30.         //通过handlerThread.getLooper()方法才能获得Looper对象  
  31.         MyHandler myHandler = new MyHandler(handlerThread.getLooper());  
  32.         Message msg  = myHandler.obtainMessage();  
  33.         msg.sendToTarget();  //发送消息的方式  
  34.     }  
  35.       
  36.     //创建一个线程对象  
  37.     Runnable r = new Runnable() {  
  38.           
  39.         @Override  
  40.         public void run() {  
  41.             Log.e("handler-->" ,Thread.currentThread().getId() + "");  
  42.             Log.e("handlername-->" ,Thread.currentThread().getName() + "");  
  43.             try{  
  44.                 Thread.sleep(10*1000);  
  45.             }catch(InterruptedException e){  
  46.                 e.printStackTrace();  
  47.             }  
  48.         }  
  49.     };  
  50.       
  51.     //创建Handler对象  
  52.     class MyHandler extends Handler{  
  53.           
  54.         public MyHandler(){  
  55.               
  56.         }  
  57.           
  58.         public MyHandler(Looper looper){  
  59.             super(looper);  
  60.         }  
  61.   
  62.         @Override  
  63.         public void handleMessage(Message msg) {  
  64.             Log.e("handler-->" ,Thread.currentThread().getId() + "");  
  65.             Log.e("handler-->" ,Thread.currentThread().getName() + "");  
  66.         }  
  67.           
  68.     }  
  69. }  

package com.andyidea.handlerdemo2; import android.app.Activity; import android.os.Bundle; import android.os.Handler; import android.os.HandlerThread; import android.os.Looper; import android.os.Message; import android.util.Log; public class HandlerActivity extends Activity { private Handler handler = new Handler(); @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); //handler.post(r); setContentView(R.layout.main); Log.e("activity-->" ,Thread.currentThread().getId() + ""); Log.e("activityname-->" ,Thread.currentThread().getName() + ""); //与主线程同一线程 handler.post(r); //与主线程不同线程 HandlerThread handlerThread = new HandlerThread("handler_thread"); handlerThread.start(); //HandlerThread线程对象要启动后, //通过handlerThread.getLooper()方法才能获得Looper对象 MyHandler myHandler = new MyHandler(handlerThread.getLooper()); Message msg = myHandler.obtainMessage(); msg.sendToTarget(); //发送消息的方式 } //创建一个线程对象 Runnable r = new Runnable() { @Override public void run() { Log.e("handler-->" ,Thread.currentThread().getId() + ""); Log.e("handlername-->" ,Thread.currentThread().getName() + ""); try{ Thread.sleep(10*1000); }catch(InterruptedException e){ e.printStackTrace(); } } }; //创建Handler对象 class MyHandler extends Handler{ public MyHandler(){ } public MyHandler(Looper looper){ super(looper); } @Override public void handleMessage(Message msg) { Log.e("handler-->" ,Thread.currentThread().getId() + ""); Log.e("handler-->" ,Thread.currentThread().getName() + ""); } } }

运行程序后,我们通过Logcat控制台的log.e输出可以看到结果如下:

通过验证,可以看到如果只是通过Handler对象的post()方法,只是把一个子线程对象放到主线程队列中,并不是开启了一个子线程。通过HandlerThread对象方式才是开启了一个新的线程去执行程序。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值