常用的两种handler调用方法

2024-09-05 07:58

本文主要是介绍常用的两种handler调用方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、Handler、Thread、HandlerThread三者之间的关系如下:

1、Handler:在android中负责发送和处理消息,通过它可以实现其他支线线程与主线程之间的消息通讯。

2、Thread:Java进程中执行运算的最小单位,亦即执行处理机调度的基本单位。某一进程中一路单独运行的程序。

3、HandlerThread:一个继承自Thread的类HandlerThread。
二、Handler跟Thread结合使用,首先在FirstHandler.java类中创建要在界面调用的getSum方法,具体实现如下:
 

package com.example.handlerdemo;import android.os.Handler;
import android.os.Message;public class FirstHandler {private Handler mHandler;public static final int ADD_FLAG = 1;public void setHandler(Handler handler){mHandler = handler;}public synchronized void getSum(final int a, final int b){Logger.d("a = " + a + " ,b = " + b);class AddRunThread implements Runnable{@Overridepublic void run() {Logger.d("FirstHandler ThreadId = " + Thread.currentThread().getId());int sum = add(a, b); try {Thread.sleep(6 * 1000); //模拟加法运算是一个非常复杂的运算,所以要开线程} catch (InterruptedException e) {e.printStackTrace();}String result = "result = " + sum;Message message = mHandler.obtainMessage();message.what = ADD_FLAG;message.obj = result;mHandler.sendMessage(message); //在新的线程中进行发送Message消息}}Thread addRun = new Thread(new AddRunThread());addRun.setPriority(Thread.MAX_PRIORITY); //设置线程优先级addRun.start();//线程启动}/*** 实现加法运算*/private int add(int a, int b){return a + b;}
}

当调用 getSum方法的时候会先启动addRun线程,addRun线程启动后就会执行AddRunThread类里面的run方法,run方法里面执行了一个加法运算的add方法,然后会执行Thread.sleep(6 * 1000)来模拟耗时的操作。加法运算add方法实行结束后会通过mHandler.sendMessage(message)来通知界面更新信息。接下来看一下界面MainActivity的实现,具体如下:

package com.example.handlerdemo;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.TextView;public class MainActivity extends Activity {private TextView textView;private FirstHandler firstHandler;private SecondHandler secondHandler;private String result;private Handler handler = new Handler(){public void handleMessage(android.os.Message msg) {Logger.d("Handler ThreadId = " + Thread.currentThread().getId());switch (msg.what) {case FirstHandler.ADD_FLAG:Logger.d("FirstHandler.ADD_FLAG");textView.setText((String) msg.obj); //更新加法运算后的结果break;case SecondHandler.UI_REFLESH:Logger.d("SecondHandler.UI_REFLESH"); textView.setText((String) msg.obj); //更新减法运算后的结果break;default:break;}};};@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);Logger.d("main ThreadId = " + Thread.currentThread().getId());setContentView(R.layout.activity_main);textView = (TextView) findViewById(R.id.tx_show);firstHandler = new FirstHandler();secondHandler = new SecondHandler();}/** handler跟Thread结合使用的测试方法*/public void addCheck(View view){Logger.d();firstHandler.setHandler(handler); //设置Handler对象实例firstHandler.getSum(3000, 4600);  //调用FirstHandler类的getSum方法}/** handler跟HandlerThread结合使用的测试方法*/public void decreaseClick(View view){Logger.d();secondHandler.getSerialThread().post(new Runnable() {@Overridepublic void run() {Logger.d("SecondHandler ThreadId = " + Thread.currentThread().getId());int sum = decrease(300, 180);try {Thread.sleep(8 * 1000); //模拟减法运算是一个非常复杂的运算,所以要开线程} catch (InterruptedException e) {e.printStackTrace();}result = "result2 = " + sum;Message message = new Message();message.what = SecondHandler.UI_REFLESH;message.obj = result;handler.sendMessage(message); //通过主线程更新UI}});}//减法运算private int decrease(int a, int b){return a - b;}}

 可以看到在handleMessage中进行了消息处理,当msg.what等于FirstHandler.ADD_FLAG时,就会将加法运算的结果设置到textView中展示。

 

 

三、handler跟HandlerThread结合使用。代码逻辑主要在SecondHandler.java类中实现,具体如下:

package com.example.handlerdemo;import android.os.Handler;
import android.os.HandlerThread;
import android.os.Looper;public class SecondHandler {public Handler uiHandler = new Handler(Looper.getMainLooper()); //main主线程,由于更新UI更新private HandlerThread mHandlerThread;private Handler handler;public static final int UI_REFLESH = 0X02;//设置handler对象关联public void setUIhandler(Handler handler){uiHandler = handler;}/***获取获得Looper对象的Handler实例*/public synchronized Handler getSerialThread(){if (null == mHandlerThread){HandlerThread thread = new HandlerThread("work_thread");//工作线程,用于耗时的操作,不能用于更新UIthread.start();handler = new Handler(thread.getLooper()){};}return handler;}
}

通过 new Handler(thread.getLooper())来建立handler跟HandlerThread的联系。接下来看一下具体怎么调用getSerialThread()方法,如下:

 /** handler跟HandlerThread结合使用的测试方法*/public void decreaseClick(View view){Logger.d();secondHandler.getSerialThread().post(new Runnable() {@Overridepublic void run() {Logger.d("SecondHandler ThreadId = " + Thread.currentThread().getId());int sum = decrease(300, 180);try {Thread.sleep(8 * 1000); //模拟减法运算是一个非常复杂的运算,所以要开线程} catch (InterruptedException e) {e.printStackTrace();}result = "result2 = " + sum;Message message = new Message();message.what = SecondHandler.UI_REFLESH;message.obj = result;handler.sendMessage(message); //通过主线程更新UI}});}//减法运算private int decrease(int a, int b){return a - b;}

首先通过 getSerialThread()方法获取secondHandler对象,然后调用了post方法,在run方法中执行了减法decrease(300, 180)运算,减法运算结束后会调用一个休眠函数Thread.sleep(8 * 1000),休眠8秒来模拟是耗时操作,休眠结束后会通过主线程的handler来通知界面更新减法的结果。运行结果如下:

 代码参考:https://github.com/gunder1129/android-tool/commit/219738d73f22763ab531800e423a66c8d3e72860

参考博客:https://blog.csdn.net/weixin_41101173/article/details/79687313

 

 

这篇关于常用的两种handler调用方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1138381

相关文章

Java controller接口出入参时间序列化转换操作方法(两种)

《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

Python中pywin32 常用窗口操作的实现

《Python中pywin32常用窗口操作的实现》本文主要介绍了Python中pywin32常用窗口操作的实现,pywin32主要的作用是供Python开发者快速调用WindowsAPI的一个... 目录获取窗口句柄获取最前端窗口句柄获取指定坐标处的窗口根据窗口的完整标题匹配获取句柄根据窗口的类别匹配获取句

Java 中的 @SneakyThrows 注解使用方法(简化异常处理的利与弊)

《Java中的@SneakyThrows注解使用方法(简化异常处理的利与弊)》为了简化异常处理,Lombok提供了一个强大的注解@SneakyThrows,本文将详细介绍@SneakyThro... 目录1. @SneakyThrows 简介 1.1 什么是 Lombok?2. @SneakyThrows

判断PyTorch是GPU版还是CPU版的方法小结

《判断PyTorch是GPU版还是CPU版的方法小结》PyTorch作为当前最流行的深度学习框架之一,支持在CPU和GPU(NVIDIACUDA)上运行,所以对于深度学习开发者来说,正确识别PyTor... 目录前言为什么需要区分GPU和CPU版本?性能差异硬件要求如何检查PyTorch版本?方法1:使用命

Qt实现网络数据解析的方法总结

《Qt实现网络数据解析的方法总结》在Qt中解析网络数据通常涉及接收原始字节流,并将其转换为有意义的应用层数据,这篇文章为大家介绍了详细步骤和示例,感兴趣的小伙伴可以了解下... 目录1. 网络数据接收2. 缓冲区管理(处理粘包/拆包)3. 常见数据格式解析3.1 jsON解析3.2 XML解析3.3 自定义

SpringMVC 通过ajax 前后端数据交互的实现方法

《SpringMVC通过ajax前后端数据交互的实现方法》:本文主要介绍SpringMVC通过ajax前后端数据交互的实现方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价... 在前端的开发过程中,经常在html页面通过AJAX进行前后端数据的交互,SpringMVC的controll

Java中的工具类命名方法

《Java中的工具类命名方法》:本文主要介绍Java中的工具类究竟如何命名,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录Java中的工具类究竟如何命名?先来几个例子几种命名方式的比较到底如何命名 ?总结Java中的工具类究竟如何命名?先来几个例子JD

Spring Security自定义身份认证的实现方法

《SpringSecurity自定义身份认证的实现方法》:本文主要介绍SpringSecurity自定义身份认证的实现方法,下面对SpringSecurity的这三种自定义身份认证进行详细讲解,... 目录1.内存身份认证(1)创建配置类(2)验证内存身份认证2.JDBC身份认证(1)数据准备 (2)配置依

C#使用StackExchange.Redis实现分布式锁的两种方式介绍

《C#使用StackExchange.Redis实现分布式锁的两种方式介绍》分布式锁在集群的架构中发挥着重要的作用,:本文主要介绍C#使用StackExchange.Redis实现分布式锁的... 目录自定义分布式锁获取锁释放锁自动续期StackExchange.Redis分布式锁获取锁释放锁自动续期分布式