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

相关文章

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 第一节:嵌入方程的类比与核心概念【尽可能通俗】 嵌入方程可以被看作是自然语言处理中的“翻译机”,它将文本中的单词或短语转换成计算机能够理解的数学形式,即向量。 正如翻译机将一种语言

[Linux]:进程(下)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:Linux学习 贝蒂的主页:Betty’s blog 1. 进程终止 1.1 进程退出的场景 进程退出只有以下三种情况: 代码运行完毕,结果正确。代码运行完毕,结果不正确。代码异常终止(进程崩溃)。 1.2 进程退出码 在编程中,我们通常认为main函数是代码的入口,但实际上它只是用户级

android-opencv-jni

//------------------start opencv--------------------@Override public void onResume(){ super.onResume(); //通过OpenCV引擎服务加载并初始化OpenCV类库,所谓OpenCV引擎服务即是 //OpenCV_2.4.3.2_Manager_2.4_*.apk程序包,存

flume系列之:查看flume系统日志、查看统计flume日志类型、查看flume日志

遍历指定目录下多个文件查找指定内容 服务器系统日志会记录flume相关日志 cat /var/log/messages |grep -i oom 查找系统日志中关于flume的指定日志 import osdef search_string_in_files(directory, search_string):count = 0

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

Android 10.0 mtk平板camera2横屏预览旋转90度横屏拍照图片旋转90度功能实现

1.前言 在10.0的系统rom定制化开发中,在进行一些平板等默认横屏的设备开发的过程中,需要在进入camera2的 时候,默认预览图像也是需要横屏显示的,在上一篇已经实现了横屏预览功能,然后发现横屏预览后,拍照保存的图片 依然是竖屏的,所以说同样需要将图片也保存为横屏图标了,所以就需要看下mtk的camera2的相关横屏保存图片功能, 如何实现实现横屏保存图片功能 如图所示: 2.mtk