Android Service进程间双向通信之Messenger(系列4)

2023-12-22 22:32

本文主要是介绍Android Service进程间双向通信之Messenger(系列4),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Android Service进程间双向通信之Messenger(系列4)

附录文章2虽然利用Service的Binder、bindService这些机制实现了Android Service与其他组件的相互通信,但实现手段并不唯一,Android体系架构中还有一个解决方案:利用Android Messenger实现Service进程间双向通信。


先丢出代码。


先写一个MyService.Java,继承自Service:

[java]  view plain copy
  1. package zhangphil.service;  
  2.   
  3. import android.app.Service;  
  4. import android.content.Intent;  
  5. import android.os.Handler;  
  6. import android.os.IBinder;  
  7. import android.os.Message;  
  8. import android.os.Messenger;  
  9. import android.util.Log;  
  10.   
  11. public class MyService extends Service {  
  12.   
  13.     private Messenger messenger;  
  14.   
  15.     @Override  
  16.     public void onCreate() {  
  17.         Log.d(this.getClass().getName(), "onCreate");  
  18.   
  19.         Handler handler = new Handler() {  
  20.             public void handleMessage(Message msg) {  
  21.                 Log.d(this.getClass().getName(), msg.what + ":" + msg.obj);  
  22.   
  23.                 // 收到来自于Activity的消息后立即响应回复一条消息。  
  24.                 sendMessageToActivity(msg.replyTo);  
  25.             }  
  26.         };  
  27.   
  28.         messenger = new Messenger(handler);  
  29.     }  
  30.   
  31.     @Override  
  32.     public int onStartCommand(Intent intent, int flags, int startId) {  
  33.         return super.onStartCommand(intent, flags, startId);  
  34.     }  
  35.   
  36.     private void sendMessageToActivity(Messenger mMessenger) {  
  37.         Message msg = Message.obtain();  
  38.         msg.what = 0x09;  
  39.         msg.obj = "hello,i'm from Service !";  
  40.   
  41.         try {  
  42.             mMessenger.send(msg);  
  43.         } catch (Exception e) {  
  44.             e.printStackTrace();  
  45.         }  
  46.     }  
  47.   
  48.     @Override  
  49.     public IBinder onBind(Intent intent) {  
  50.         return messenger.getBinder();  
  51.     }  
  52.   
  53.     // @Override  
  54.     // public boolean onUnbind(Intent intent) {  
  55.     // Log.d(this.getClass().getName(), "onUnbind");  
  56.     // return super.onUnbind(intent);  
  57.     // }  
  58.     //  
  59.     // @Override  
  60.     // public void onDestroy() {  
  61.     // Log.d(this.getClass().getName(), "onDestroy");  
  62.     // }  
  63. }  



测试的Activity MainActivity.java:

[java]  view plain copy
  1. package zhangphil.service;  
  2.   
  3. import android.app.Activity;  
  4. import android.app.Service;  
  5. import android.content.ComponentName;  
  6. import android.content.Intent;  
  7. import android.content.ServiceConnection;  
  8. import android.os.Bundle;  
  9. import android.os.Handler;  
  10. import android.os.IBinder;  
  11. import android.os.Message;  
  12. import android.os.Messenger;  
  13. import android.util.Log;  
  14. import android.view.View;  
  15. import android.widget.Button;  
  16.   
  17. public class MainActivity extends Activity {  
  18.   
  19.     private ServiceConnection sc = null;  
  20.     private Messenger sender, receiver;  
  21.   
  22.     @Override  
  23.     protected void onCreate(Bundle savedInstanceState) {  
  24.         super.onCreate(savedInstanceState);  
  25.         setContentView(R.layout.activity_main);  
  26.   
  27.         bindMyService();  
  28.   
  29.         Handler handler = new Handler() {  
  30.             public void handleMessage(Message msg) {  
  31.                 Log.d(this.getClass().getName(), msg.what + ":" + msg.obj);  
  32.             }  
  33.         };  
  34.   
  35.         receiver = new Messenger(handler);  
  36.   
  37.         Button sendToService = (Button) findViewById(R.id.sendToService);  
  38.         sendToService.setOnClickListener(new View.OnClickListener() {  
  39.   
  40.             @Override  
  41.             public void onClick(View v) {  
  42.                 sendMessageToService(sender);  
  43.             }  
  44.         });  
  45.     }  
  46.   
  47.     private void sendMessageToService(Messenger messenger) {  
  48.         Message msg = Message.obtain();  
  49.         msg.what = 0x08;  
  50.         msg.obj = "hello,i'm from Activity !";  
  51.   
  52.         // 设置一个Messenger receiver,receiver是提供给Service使用来给Activity响应的目标。  
  53.         msg.replyTo = receiver;  
  54.   
  55.         try {  
  56.             messenger.send(msg);  
  57.         } catch (Exception e) {  
  58.             e.printStackTrace();  
  59.         }  
  60.     }  
  61.   
  62.     @Override  
  63.     protected void onDestroy() {  
  64.         super.onDestroy();  
  65.         unbindMyService();  
  66.         Log.d(this.getClass().getName(), "onDestroy");  
  67.     }  
  68.   
  69.     // private void startMyAppService() {  
  70.     // Intent intent = new Intent(this, MyService.class);  
  71.     // this.startService(intent);  
  72.     // }  
  73.   
  74.     private void bindMyService() {  
  75.         sc = new ServiceConnection() {  
  76.   
  77.             @Override  
  78.             public void onServiceConnected(ComponentName name, IBinder binder) {  
  79.                 Log.d(this.getClass().getName(), "onServiceConnected");  
  80.   
  81.                 sender = new Messenger(binder);  
  82.             }  
  83.   
  84.             @Override  
  85.             public void onServiceDisconnected(ComponentName name) {  
  86.                 Log.d(this.getClass().getName(), "onServiceDisconnected");  
  87.             }  
  88.         };  
  89.   
  90.         Intent intent = new Intent(this, MyService.class);  
  91.         this.bindService(intent, sc, Service.BIND_AUTO_CREATE);  
  92.     }  
  93.   
  94.     // private void stopMyService() {  
  95.     // Intent intent = new Intent(this, MyService.class);  
  96.     // boolean bool = this.stopService(intent);  
  97.     // }  
  98.   
  99.     private void unbindMyService() {  
  100.         if (sc != null)  
  101.             this.unbindService(sc);  
  102.     }  
  103. }  

以下是代码说明。
先说一下代码要实现的简单目的:首先绑定后台Service:MyService,然后Activity向后台Service发送一条的简单字符串消息;当后台的Service收到消息后,立即响应再发给前台的Activity一个字符串消息。
使用Android Messenger实现Service与其他组件之间的双向通信需要注意几点内容:
(A)和参考文章2中的Service不同的是,这一次,public IBinder onBind(Intent intent)返回的是不是自己写的Binder,而是一个从Messenger获得的Binder。构造Messenger时候给其传递一个handler,handler用于接受发送给Service的消息Message(在本例中是Activity发给Service的消息)。
(B)在本例中,后台Service与前台Activity通信‘由后向前’的通信关键是:在Activity给Service发送消息时候,设置一个replyTo的Messenger,此replyTo字段中的Messenger将被消息接受者(Service)捕获提取,然后作为目标靶Messenger发消息,从而实现Service接受消息后,可以用replyTo字段中的Messenger返还一个消息给Activity。
(C)和参考文章2类似,前台的Activity要绑定Service,ServiceConnection绑定后在onServiceConnected中获得一个可以给后台Service发消息的Messenger(在本例中是Activity中的sender)。此Messenger即是和Activity和Service通信的桥梁。
(D)前台Activity如果要打算接受来自于后台Service的消息,那么就必须再创建一个额外的Messenger(在本例中是Activity中的receiver),同样需要创建一个Activity自己的Handler,handler传递给receiver,Activity的Handler将接收来自于Service发送过来的消息。
(E)若要实现双向通信,务必记住,前台Activity用(C)阶段获得的Messenger sender给后台的Service发送消息时候,务必将Activity中的Messenger receiver赋值到Message的replyTo,传递给后台的Service。即msg. msg.replyTo = receiver


这篇关于Android Service进程间双向通信之Messenger(系列4)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

一文解密Python进行监控进程的黑科技

《一文解密Python进行监控进程的黑科技》在计算机系统管理和应用性能优化中,监控进程的CPU、内存和IO使用率是非常重要的任务,下面我们就来讲讲如何Python写一个简单使用的监控进程的工具吧... 目录准备工作监控CPU使用率监控内存使用率监控IO使用率小工具代码整合在计算机系统管理和应用性能优化中,监

Linux进程CPU绑定优化与实践过程

《Linux进程CPU绑定优化与实践过程》Linux支持进程绑定至特定CPU核心,通过sched_setaffinity系统调用和taskset工具实现,优化缓存效率与上下文切换,提升多核计算性能,适... 目录1. 多核处理器及并行计算概念1.1 多核处理器架构概述1.2 并行计算的含义及重要性1.3 并

Linux下进程的CPU配置与线程绑定过程

《Linux下进程的CPU配置与线程绑定过程》本文介绍Linux系统中基于进程和线程的CPU配置方法,通过taskset命令和pthread库调整亲和力,将进程/线程绑定到特定CPU核心以优化资源分配... 目录1 基于进程的CPU配置1.1 对CPU亲和力的配置1.2 绑定进程到指定CPU核上运行2 基于

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

Javaee多线程之进程和线程之间的区别和联系(最新整理)

《Javaee多线程之进程和线程之间的区别和联系(最新整理)》进程是资源分配单位,线程是调度执行单位,共享资源更高效,创建线程五种方式:继承Thread、Runnable接口、匿名类、lambda,r... 目录进程和线程进程线程进程和线程的区别创建线程的五种写法继承Thread,重写run实现Runnab

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

Android DataBinding 与 MVVM使用详解

《AndroidDataBinding与MVVM使用详解》本文介绍AndroidDataBinding库,其通过绑定UI组件与数据源实现自动更新,支持双向绑定和逻辑运算,减少模板代码,结合MV... 目录一、DataBinding 核心概念二、配置与基础使用1. 启用 DataBinding 2. 基础布局