花样使用Handler与源码分析

前几天在跟公司大佬讨论一个问题时,看到他使用Handler的一种方式,旁边的同事在说:以前不是这么用的啊。这个问题引发了我的好奇,虽然当时翻清楚道理了,但是还是想给大家分享一下。

Handler在之前也说到过他的使用以及源码分析,而且相信大家都知道如何使用它,最常见的使用方法恐怕就是下面这种了:

Handler handler = new Handler(){
  @Override
  public void handleMessage(Message msg) {
    super.handleMessage(msg);
  }
};

这种情况会有一个问题:我们都知道Handler是可以用在子线程给主线程更新的,当子线程给主线程回调时,主线程中的Handler通过接收发送过来的对应消息,去执行对应的任务。而对于上面这个Handler对象,如果他是主线程中的,那么我们子线程中需要拿到主线程的这个Handler对象。

  final Handler handler = new Handler(){
    @Override
    public void handleMessage(Message msg) {
      super.handleMessage(msg);
    }
  };
  new Thread(new Runnable() {
    @Override
    public void run() {
      handler.sendMessage(new Message());
    }
  }).start();

但是上面这种写法实在是太不好看了,而且handler还是一个局部变量,在其他方法中也无法使用。

  Handler handler;
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    Log.e(TAG, "onCreate: " );
    handler = new Handler(){
      @Override
      public void handleMessage(Message msg) {
        super.handleMessage(msg);
      }
    };
    new Thread(new Runnable() {
      @Override
      public void run() {
        handler.sendMessage(new Message());
      }
    }).start();
  }

这个看上去应该就好多了,可能也是大多数人的一种写法。

其实说白了,如果说我们要在子线程中给主线程/相应线程回调,那么一定要拿到主线程中的Handler的索引。这么说就很直接了,可能有些情况下无法拿到主线程/相应线程的Handler,或者拿到的方法很麻烦:

public class MyService extends Service {
  @Nullable
  @Override
  public IBinder onBind(Intent intent) {
    return null;
  }
  @Override
  public void onCreate() {
    super.onCreate();
    new Thread(new Runnable() {
      @Override
      public void run() {
        //执行相关的耗时操作,然后结束后通过Handler回调给主线程
      }
    }).start();
  }
}

现在我有这样的需求:创建一个服务,在服务中开启一个子线程执行耗时操作,当执行完毕后回调在主线程中相应。这种情况下想要拿到主线程的Handler对象也不是不可以,方法还是有很多,把主线程的handler写成static、创建类继承Handler并且序列化,然后通过intent传入.....可能还有其他的一些方法,但是就目前的这些情况来看,貌似都不是很友好。下面给大家带来一种比较优雅且方便的方法:

public class MyService extends Service {
  @Nullable
  @Override
  public IBinder onBind(Intent intent) {
    return null;
  }
  @Override
  public void onCreate() {
    super.onCreate();
    new Thread(new Runnable() {
      @Override
      public void run() {
        /*执行相关的耗时操作,然后结束后通过Handler回调给主线程*/
        new Handler(Looper.getMainLooper()).sendMessage(new Message());
      }
    }).start();
  }
}

只有一句话:又方便看着又舒服。

(可能有的朋友不知道Looper是什么,本人之前写过一篇Handler的文章对Looper有所介绍,希望对大家有所帮助:https://www.jb51.net/article/55386.htm)

通过Looper.getMainLooper方法,可以获取到主线程的Looper对象.

虽然之前说我们需要主线程中创建的Handler,其实严格的说是不对的。究其根本是因为主线程已经为他自己加载了mainLooper,而我们在主线程中new Handler,会默认获取主线程的Looper引用。

  public static void main(String[] args) {
    //pass
    Looper.prepareMainLooper();
    //pass
    Looper.loop();
    throw new RuntimeException("Main thread loop unexpectedly exited");
  }
  public Handler(Callback callback, boolean async) {
    //pass
    mLooper = Looper.myLooper();//在主线程中new的Handler获取到的looper就是主线程的mainLooper
    if (mLooper == null) {
      throw new RuntimeException(
        "Can't create handler inside thread that has not called Looper.prepare()");
    }
    //pass
  }

现在看来就很明确了,在主线程中创建Handler只是个幌子,真正在背后操纵一切的其实是looper对象。所以只需要让Handler的mLooper引用获取到主线程的引用就好了。

而且Looper.getMainLooper方法是外部可见的,大胆猜测这个方法就是为了这种方便的写法而存在的。我们可以通过这个方法获取到主线程的looper,让他实现主线程中接收回调。

  public Handler(Looper looper, Callback callback, boolean async) {
    mLooper = looper;
    mQueue = looper.mQueue;
    mCallback = callback;
    mAsynchronous = async;
  }

但是注意我们上述的这种写法:new Handler(....).sendMessage

这种写法不管你怎么去实现,他无法在主线程得到回应(是给大家挖了个坑哈哈),原因很简单:没有重写Handler.handlerMessage方法。

在使用Handler接受消息时有三种方式:

  • 重写Handler.handlerMessage方法,在该方法中接收
  • 在Handler构造器中实现Callback接口,在回调接口中接收
  • 不做任何处理,但是使用post方式发送消息。

在之前我们Handler接收消息见到的几乎都是handleMessage方法,其实这只是其中一种方法,在执行该方法之前会有一个分发的方法dispatchMessage:

  /**
   * Handle system messages here.
   **/
  public void dispatchMessage(Message msg) {
    if (msg.callback != null) {//msg中的callback,这个是通过post方法自己封装的msg(自行查源码),优先级是最高的
      handleCallback(msg);
    } else {//或者在构造器中实现Handler的Callback接口,这个优先级第二
      if (mCallback != null) {
        if (mCallback.handleMessage(msg)) {
          return;
        }
      }
      handleMessage(msg);//这才是我们之前最常用的方法,最低的优先级
    }
  }

可以看到在handlerMessage方法之前还有两种回调的方法。在上述案例中我们并没有重写第三种方法,所以对于在子线程中匿名使用Handler的情况,我们可以采取上述两种方案。代码就不写了,大家都是聪明人。

好了关于Handler 的更多使用就到这里了,喜欢的朋友希望多多支持。有不同意见和理解的希望评论区多多交流。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对我们的支持。如果你想了解更多相关内容请查看下面相关链接

(0)

相关推荐

  • .Net WebApi消息拦截器之MessageHandler的示例

    消息拦截器是一个类,接收 HTTP request并返回 HTTP response,Message handler 继承自抽象类 HttpMessageHandler 可以自定义 MessageHandler,消息拦截器的作用如: 读取或更改请求头 request headers 添加 response headers 在到达 controller 之前,进行参数验证 自定义 Message Handlers 自定义 MessageHandler 需要继承 System.Net.Http.De

  • Spring MVC学习教程之RequestMappingHandlerMapping匹配

    前言 对于RequestMappingHandlerMapping,使用Spring的同学基本都不会陌生,该类的作用有两个: 通过request查找对应的HandlerMethod,即当前request具体是由Controller中的哪个方法进行处理: 查找当前系统中的Interceptor,将其与HandlerMethod封装为一个HandlerExecutionChain. 本文主要讲解RequestMappingHandlerMapping是如何获取HandlerMethod和Interc

  • 深入Android HandlerThread 使用及其源码完全解析

    关联篇:深入Android的消息机制源码详解-Handler,MessageQueue与Looper关系 关联篇:Handler内存泄漏及其解决方案 本篇我们将来给大家介绍HandlerThread这个类,以前我们在使用线程执行一个耗时任务时总会new一个Thread的线程去跑,当任务执行完后,线程就会自动被销毁掉,如果又由新的任务,我们又得新建线程.....我们假设这样的一个情景,我们通过listview去加载图文列表,当我们往下滑动时,这时需要不断去请求网络资源,也就是需要不断开线程去加载

  • Python中logging.NullHandler 的使用教程

    在使用 peewee 框架时,默认是不会出现日志消息的. from peewee import Model, CharField, DateTimeField, IntegerField from peewee_mssql import MssqlDatabase db = MssqlDatabase(database='test', host='.', user='sa', password='sa') class BaseModel(Model): class Meta: database

  • 深入Android Handler,MessageQueue与Looper关系

    关联篇:HandlerThread 使用及其源码完全解析 关联篇:Handler内存泄漏详解及其解决方案 一说到Android的消息机制,自然就会联想到Handler,我们知道Handler是Android消息机制的上层接口,因此我们在开发过程中也只需要和Handler交互即可,很多人认为Handler的作用就是更新UI,这也确实没错,但除了更新UI,Handler其实还有很多其他用途,比如我们需要在子线程进行耗时的I/O操作,可能是读取某些文件或者去访问网络等,当耗时操作完成后我们可能需要在U

  • MyBatis使用自定义TypeHandler转换类型的实现方法

    MyBatis虽然有很好的SQL执行性能,但毕竟不是完整的ORM框架,不同的数据库之间SQL执行还是有差异. 笔者最近在升级 Oracle 驱动至 ojdbc 7 ,就发现了处理DATE类型存在问题.还好MyBatis提供了使用自定义TypeHandler转换类型的功能. 本文介绍如下使用 TypeHandler 实现日期类型的转换. 问题背景 项目中有如下的字段,是采用的DATE类型: birthday = #{birthday, jdbcType=DATE}, 在更新 Oracle 驱动之前

  • springmvc中RequestMappingHandlerAdapter与HttpMessageConverter的装配讲解

    一.DispatcherServlet 默认装配 RequestMappingHandlerAdapter ,而 RequestMappingHandlerAdapter 默认装配如下 HttpMessageConverter HttpMessageConverter: 2)加入jackson jar 包后,RequestMappingHandlerAdapter 装配的HttpMessageConverter 如下: 二. HttpMessageConverter 1)HttpMessageC

  • mybatis自定义类型处理器TypehHandler示例详解

    前言 当大家使用mybatis作为持久层框架时,在存储和查询数据时,只需要在mapper.xml文件中配置好对应字段的JdbcType和JavaType,mybatis就可以帮我们转化对应的类型.这背后是有mybatis内置的类型转换器做转换(可见源码TypeHandlerRegistry).但是有时候,我们会对某些字段做特殊处理,比如加密和解密.状态转换.类型转换等.这个时候我们需要自定义类型转换器. 类架构 从上面的图中可以看出MyBatis中整个类型处理器实现架构,TypeHandler接

  • Spring MVC学习教程之RequestMappingHandlerAdapter详解

    前言 RequestMappingHandlerAdapter实现了HandlerAdapter接口,顾名思义,表示handler的adapter,这里的handler指的是Spring处理具体请求的某个Controller的方法,也就是说HandlerAdapter指的是将当前请求适配到某个Handler的处理器.RequestMappingHandlerAdapter是HandlerAdapter的一个具体实现,主要用于将某个请求适配给@RequestMapping类型的Handler处理.

  • ASP.NET Core应用错误处理之ExceptionHandlerMiddleware中间件呈现“定制化错误页面”

    前言 DeveloperExceptionPageMiddleware中间件利用呈现出来的错误页面实现抛出异常和当前请求的详细信息以辅助开发人员更好地进行纠错诊断工作,而ExceptionHandlerMiddleware中间件则是面向最终用户的,我们可以利用它来显示一个友好的定制化的错误页面.按照惯例,我们还是先来看看ExceptionHandlerMiddleware的类型定义. public class ExceptionHandlerMiddleware { public Excepti

随机推荐