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

相关文章

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

python多进程实现数据共享的示例代码

《python多进程实现数据共享的示例代码》本文介绍了Python中多进程实现数据共享的方法,包括使用multiprocessing模块和manager模块这两种方法,具有一定的参考价值,感兴趣的可以... 目录背景进程、进程创建进程间通信 进程间共享数据共享list实践背景 安卓ui自动化框架,使用的是

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

C#如何优雅地取消进程的执行之Cancellation详解

《C#如何优雅地取消进程的执行之Cancellation详解》本文介绍了.NET框架中的取消协作模型,包括CancellationToken的使用、取消请求的发送和接收、以及如何处理取消事件... 目录概述与取消线程相关的类型代码举例操作取消vs对象取消监听并响应取消请求轮询监听通过回调注册进行监听使用Wa

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

【生成模型系列(初级)】嵌入(Embedding)方程——自然语言处理的数学灵魂【通俗理解】

【通俗理解】嵌入(Embedding)方程——自然语言处理的数学灵魂 关键词提炼 #嵌入方程 #自然语言处理 #词向量 #机器学习 #神经网络 #向量空间模型 #Siri #Google翻译 #AlexNet 第一节:嵌入方程的类比与核心概念【尽可能通俗】 嵌入方程可以被看作是自然语言处理中的“翻译机”,它将文本中的单词或短语转换成计算机能够理解的数学形式,即向量。 正如翻译机将一种语言