好玩的Handler

  1. Android提供了Handler和Looper来满足线程间的通信;
  2. Handler和Activity的任务栈不同,它是先进先出原则;
  3. Handler:你可以构造Handler对象来与Looper沟通,以便push新消息到MessageQueue里,或者接口Looper从MessageQueue取出的消息;
  4. Looper类用来管理特定线程内对象之间交换Message;
  5. 一个线程可以产生一个Looper对象,由他来管理此线程的MessageQueue(消息队列);
  6. MessageQueue:用来存放线程放入的消息;
  7. 每一个消息都需要制定的Handler来处理,通过Handler创建消息便可以完成此功能.Android引入了消息池.Handler创建消息时首先查询消息池中是否有消息存在,如果有,则直接取出,如果没有,则重新初始化一个消息实例.
  8. 使用消息池的好处是:消息不被使用时,并不作为垃圾回收,而是放入消息池中,可供下次Handler创建消息时使用.消息池提高了消息对象的复用,减少系统垃圾回收的次数.Message.obtain()来获取消息,最大数为50;


  9.  


  10.  综上所述:这就是一个标准的的异步操作,就像我们寄信一样,我们只负责写好信(Message)通过邮递员(Handler)放入到邮箱(MessageQueue)中,由工作人员(Looper)去循环查询,再由邮递员处理(Handler)处理这些消息;
  11. 应用场景:两秒后打开一个Activity

    1. //延迟两秒跳转
    2. newHandler().postDelayed(newRunnable(){
    3. @Override
    4. publicvoid run(){
    5. Intent intent=newIntent(MainActivity.this,TestActivity.class);
    6. startActivity(intent);
    7. }
    8. },2000);

  12. 先来看一个简单的消息吧

    1. privateProgressBar mProgressBar;
    2. privateint i =0;
    3. privateHandler mHandler =newHandler(){ // 创建Handle
    4. @Override
    5. publicvoid handleMessage(Message msg){
    6. super.handleMessage(msg);
    7. Log.i("-mHandler->",i+"");
    8. mProgressBar.setProgress(i);
    9. }
    10. };
    11. privateRunnable runnable =newRunnable(){
    12. @Override
    13. publicvoid run(){
    14. Log.i("-Runnable->",i+"");
    15. i +=10;
    16. // 要做的事情,这里再次调用此Runnable对象,以实现每两秒实现一次的定时器操作
    17. mHandler.postDelayed(runnable,2000); // 定时器
    18. mHandler.sendMessageDelayed(Message.obtain(),0);// 发送消息才会触发重写的handleMessage方法
    19. }
    20. };
    21. @Override
    22. protectedvoid onCreate(@NullableBundle savedInstanceState){
    23. super.onCreate(savedInstanceState);
    24. setContentView(R.layout.activity_handler);
    25. mProgressBar =(ProgressBar) findViewById(R.id.pb_handler);
    26. mHandler.post(runnable);// 开始执行线程
    27. // runnable.run(); // 也可以用这个来开始线程
    28. }
    29. /**
    30. * 删除的时候停止线程操作
    31. */
    32. @Override
    33. protectedvoid onDestroy(){
    34. super.onDestroy();
    35. mHandler.removeCallbacks(runnable);
    36. }
    37. }

  13. 获取Message的两种方法
    1. // 两种获取Message的方法 
    2. // Returns a new Message from the global message pool.
    3. mHandler.obtainMessage()和Message.obtain()
    1. // 两种方法其实是一样的,只不过一个是通过Handler获取,一个是通过Message的静态方法获得,查询Handler的源码会发现,obtainMessage的方法构成:
    2. publicfinalMessage obtainMessage()
    3. {
    4. returnMessage.obtain(this);
    5. }
  14. 通过查询源码后发现,Message提供了诸如以下的变量

    1. Message的recycleUnchecked()方法
    2. void recycleUnchecked(){
    3. // Mark the message as in use while it remains in the recycled object pool.
    4. // Clear out all other details.
    5. flags = FLAG_IN_USE; //int
    6. what =0; // int
    7. arg1 =0; // int
    8. arg2 =0; // int
    9. obj =null; // Object
    10. replyTo =null; // Messenger 信使,信差
    11. sendingUid =-1;
    12. when =0; // long
    13. target =null; // Handler
    14. callback =null; // Runable
    15. data =null; // Bundle
    16. synchronized(sPoolSync){
    17. if(sPoolSize < MAX_POOL_SIZE){
    18. next = sPool;
    19. sPool =this;
    20. sPoolSize++;
    21. }
    22. }
    23. }

    使用系统变量的好处是可以大大减少系统的消耗;所以更新进度条的代码应修改为

    1. mProgressBar.setProgress(msg.arg1);
    1. privateRunnable runnable =newRunnable(){
    2. @Override
    3. publicvoid run(){
    4. Log.i("-Runnable->", i +"");
    5. i +=3;
    6. // 要做的事情,这里再次调用此Runnable对象,以实现每两秒实现一次的定时器操作
    7. mHandler.postDelayed(runnable,300);
    8. Message msg = mHandler.obtainMessage();
    9. msg.arg1 +=i;
    10. mHandler.sendMessage(msg);// 发送消息才会触发重写的handleMessage方法
    11. // mHandler.sendMessageDelayed(Message.obtain(),0); // 发送消息才会触发重写的handleMessage方法
    12. }
    13. };

  15. 重新优化下Handler
    1. privateHandler mHandler =newHandler(){
    2. @Override
    3. publicvoid handleMessage(Message msg){
    4. super.handleMessage(msg);
    5. //Log.i("-mHandler->", i + "");
    6. if(msg.arg1 >100){
    7. mHandler.removeCallbacks(runnable);
    8. }else{
    9. mProgressBar.setProgress(msg.arg1);
    10. }
    11. }
    12. };

来自为知笔记(Wiz)

时间: 2024-12-22 17:12:00

好玩的Handler的相关文章

Handler使用

1.为什么要使用Handler机制呢? 一般更新UI,我们都使用线程防止主线程的卡顿,但是我们都知道,java是线程不安全的,很容易造成不同的线程操作同一个view产生混乱.我们无法做到给每个线程,每个view对象都加一个锁,所以谷歌提供了一个给我们handler,通过looper这样的不断压栈出栈的规则,来进行线程的更新. 讲是这么讲,不在线程(非主线程)中使用Handler与主线程产生联系,压根更新不了UI(不用的话会报错的). 2.Handler的用途是什么呢? (1)更新UI (2)消息

安卓 Handler

1. 看视频有关Handler操作,没有用过,里面的老师也没有讲(新手太难了) 查了一下资料. 一.Handler的定义: 主要接受子线程发送的数据, 并用此数据配合主线程更新UI. 解释:当应用程序启动时,Android首先会开启一个主线程 (也就是UI线程) , 主线程为管理界面中的UI控件, 进行事件分发, 比如说, 你要是点击一个 Button ,Android会分发事件到Button上,来响应你的操作. 如果此时需要一个耗时的操作,例如: 联网读取数据, 或者读取本地较大的一个文件的时

Android中关于Handler的总结

一.Handler的定义: 主要接受子线程发送的数据, 并用此数据配合主线程更新UI.          解释: 当应用程序启动时,Android首先会开启一个主线程 (也就是UI线程) , 主线程为管理界面中的UI控件,进行事件分发, 比如说, 你要是点击一个 Button, Android会分发事件到Button上,来响应你的操作.  如果此时需要一个耗时的操作,例如: 联网读取数据,或者读取本地较大的一个文件的时候,你不能把这些操作放在主线程中,如果你放在主线程中的话,界面会出现假死现象,

Handler, AsyncTask用法简单示例

package com.jim.testapp; import android.app.Activity; import android.os.AsyncTask; import android.os.Handler; import android.os.Message; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.util.Log; import androi

andorid 多线程handler用法

.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent&qu

class ha_innobase: public handler

/** The class defining a handle to an Innodb table */ class ha_innobase: public handler { row_prebuilt_t* prebuilt; /*!< prebuilt struct in InnoDB, used to save CPU time with prebuilt data structures*/ THD* user_thd; /*!< the thread handle of the us

Handler和Message详解

Android开发中Handler的经典总结 当应用程序启动时,Android首先会开启一个主线程(也就是UI线程),主线程为管理界面中的UI控件,进行事件分发. 一.Handler的定义: 主要接受子线程发送的数据, 并用此数据配合主线程更新UI. 解释:当应用程序启动时,Android首先会开启一个主线程 (也就是UI线程) , 主线程为管理界面中的UI控件, 进行事件分发, 比如说, 你要是点击一个 Button ,Android会分发事件到Button上,来响应你的操作.  如果此时需要

Android Handler

1 package com.example.myapp7; 2 3 import android.support.v7.app.ActionBarActivity; 4 import android.os.Bundle; 5 import android.os.Handler; 6 import android.os.Looper; 7 import android.os.Message; 8 import android.util.Log; 9 import android.view.Menu

Android开发:Handler Runnable和Thread之间的区别和联系 应用--------------------看完本篇,从此一览无余!

http://blog.csdn.net/yanzi1225627/article/details/8582081 在java中可有两种方式实现多线程,一种是继承Thread类,一种是实现Runnable接口:Thread类是在java.lang包中定义的.一个类只要继承了Thread类同时覆写了本类中的run()方法就可以实现多线程操作了,但是一个类只能继承一个父类,这是此方法的局限. 下面看例子: package org.thread.demo; class MyThread extends