本文转自:http://www.cnblogs.com/keyindex/articles/1822463.html

前言

学习android一段时间了,为了进一步了解android的应用是如何设计开发的,决定详细研究几个开源的android应用。从一些开源应用中吸收点东西,一边进行量的积累,一边探索android的学习研究方向。这里我首先选择了jwood的Standup Timer 项目。本文将把研究的内容笔记整理,建立一个索引列表。

关键词

Android.os.Handler涉及较多的知识点,我把一些关键词列举在下面,将主要介绍Handler:
  •   android.os.Handler 、 android.os.Handler.Callback
  •   Looper、
  •   Threadle、Runnable
  •   Message、Message queue

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布局:
<?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"/>
<Buttonandroid:id="@+id/btnStartTime"android:text="开始计时"android:layout_width="80dip"android:layout_height="wrap_content" ></Button><Buttonandroid: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主要是用于查看线程是否被阻塞(阻塞时无法拖动)。

  @Overridepublicvoid 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中用以和消息发送、处理时所作的线程进行比较。
发送消息@Overridepublicvoid 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继承自TimerTaskTask updateTimerValuesTask =new Task(this);timer.schedule(updateTimerValuesTask, 1000, 1000);}//执行耗时的倒计时任务。
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();
        }@Overridepublicvoid 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;}@Overridepublicvoid run() {// TODO Auto-generated method stub
        iTask.TaskRun();}}

这是Java的回调函数的一般写法。

实现CallBack/*** 实现消息处理*/@Overridepublicboolean 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. */@Overridepublicvoid 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);}/*** 实现消息处理*/@Overridepublicboolean 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;}@Overridepublicvoid 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;}@Overridepublicvoid 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发送给消息队列,它将在到达时进行处理。
POSTpublicclass PostHandler extends Activity implements OnClickListener, Runnable {private TextView txt;private Button btnStart, btnStop;private Handler myHandler;private Timer timer;privateint total =60;@Overrideprotectedvoid 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(){@Overridepublicvoid 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);}@Overridepublicvoid onClick(View v) {switch (v.getId()) {case R.id.btnStartTime://myHandler.post(this);myHandler.postDelayed(this, 1000);break;case R.id.btnStopTime:break;}}@Overridepublicvoid 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过去,运行结果是一样的。

POSTpublicclass PostHandler extends Activity implements OnClickListener, Runnable {private TextView txt;private Button btnStart, btnStop;private Handler myHandler;private Timer timer;privateint total =60;@Overrideprotectedvoid 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(){@Overridepublicvoid 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);}@Overridepublicvoid onClick(View v) {switch (v.getId()) {case R.id.btnStartTime://myHandler.post(this);myHandler.postDelayed(this, 1000);break;case R.id.btnStopTime:break;}}@Overridepublicvoid 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方法)”。

参考文献

android学习笔记之消息机制,异步和多线程
android handler概念解释
SDK

转载于:https://www.cnblogs.com/tianyaxue/p/3149118.html

Android Handler介绍相关推荐

  1. epoll监听文件_介绍一下 Android Handler 中的 epoll 机制?

    介绍一下 Android Handler 中的 epoll 机制? 目录: IO 多路复用 select.poll.epoll 对比 epoll API epoll 使用示例 Handler 中的 e ...

  2. 详解Android Handler的使用

    我们进行Android开发时,Handler可以说是使用非常频繁的一个概念,它的用处不言而喻.本文就详细介绍Handler的基本概念和用法. Handler的基本概念           Handle ...

  3. android AsyncTask介绍(转)

    android AsyncTask介绍 AsyncTask和Handler对比 1 ) AsyncTask实现的原理,和适用的优缺点 AsyncTask,是android提供的轻量级的异步类,可以直接 ...

  4. android Handler的使用(一)

    Handler的使用(一) Handler基本概念: Handler主要用于异步消息的处理:当发出一个消息之后,首先进入一个消息队列,发送消息的函数即刻返回,而另外一个部分逐个的在消息队列中将消息取出 ...

  5. 【转】Android bluetooth介绍(三): 蓝牙扫描(scan)设备分析

    原文网址:http://blog.csdn.net/xubin341719/article/details/38584469 关键词:蓝牙blueZ  A2DP.SINK.sink_connect.s ...

  6. Android Handler讲述很详细的一篇文章

    Android中的Handler <一> Handler的定义: 主要接受子线程发送的数据, 并用此数据配合主线程更新UI. 解释: 当应用程序启动时,Android首先会开启一个主线程 ...

  7. Android Handler机制简单分析

    丨版权说明 : <Android Handler机制简单分析>于当前CSDN博客和乘月网属同一原创,转载请说明出处,谢谢. 本文一切从简,将围绕以下流程展开叙述: what why how ...

  8. Android Handler的使用!!!

    大家好我们这一节讲的是Android Handler的使用,在讲Handler之前,我们先提个小问题,就是如何让程序5秒钟更新一下Title. 首先我们看一下习惯了Java编程的人,在不知道Handl ...

  9. Android Handler与Looper原理简析

    一直感觉自己简直就是一个弱智,最近越来越感觉是这样了,真的希望自己有一天能够认同自己,认同自己. 本文转载于:https://juejin.im/post/59083d7fda2f60005d14ef ...

最新文章

  1. OpenCV4中的非典型深度学习模型
  2. C#的反射机制调用方法
  3. python3菜鸟教程中文-我的python学习方法和资源整理,干货分享
  4. java线程系列---condition的讲解
  5. qemu模拟imx6用户态环境
  6. 前端学习(2348):tabber
  7. BZOJ 4278: [ONTAK2015]Tasowanie 后缀数组 + 贪心 + 细节
  8. python爬取qq好友网络状态_Python爬虫实战----爬取QQ空间好友说说并生成词云(超详细)...
  9. css模拟title和alt的提示效果[转]
  10. Ubuntu 18.04中安装飞行模拟器FlightGear 2018.2.1
  11. Xshell4简单使用
  12. Python HackerRank 刷题 Minimum Distances
  13. Excel 查找函数
  14. 【微信小程序学习笔记02理解与初始准备】【实战天气微信小程序】
  15. 木子-后端-Java小知识点
  16. gee mysql数据库_MySQL
  17. 拆长虹iho3000_(CA版)四川长虹iho-3000t晶晨s905l-b刷全网通系统教程可救砖头
  18. mel表达式_mel语言~`详解
  19. 犹太人:赚钱能培养孩子独立责任奋进多种能力!
  20. 7-30 正常血压 (6 分)

热门文章

  1. oracle主从关系表查询,Oracle 主从表联合查询解决方法
  2. vs配置编译c语言,为 VSCode 配置 C 语言编译环境
  3. Linux系统的快照是什么,linux – 文件系统快照与简单复制文件有何不同?
  4. sketch如何做设计稿交互_当屏幕可以折叠,交互设计怎么做
  5. JAVA求是否为闰年,for-while循环,输出你好
  6. 数据结构与算法(2-1)线性表之顺序存储(顺序表)
  7. gvgai框架搭建及controller编写
  8. error LNK2019: 无法解析的外部符号,该符号在函数 _main 中被引用的解决方法
  9. vcenter用到java吗_Vijava 学习笔记之 VCenter连接
  10. Linux上部署、安装nodejs