上篇文章以线程间的通信方式Handler类结尾,服务Service还支持的进程间通信,又是具体怎么实现的呢?这就要用到加载服务一文中提到的AIDL语言规范了。

AIDL是 Android Interface Definition Language 的缩写,即Android接口定义语言,使用其定义的规范编程,可实现Android系统上不同进程间的通信。官网ADIL概述中以服务端和客户端通信为例做了大致讲述。与线程间的通信类似,不同进程间的通信也是分为通信消息内容、消息发送方、消息接收方三个部分的,其中的消息内容也就是AIDL支持的数据类型;而由于服务已经位于当前应用程序所在的默认进程,所以服务中所实现的接口内容即对接收消息的处理,也就是消息接收方;相对的,其他需要发送消息的进程都可以看做是客户端进程,只要绑定服务之后都可以调用服务的接口方法,也就是消息发送方。

确定通信的消息内容

在AndroidStudio中,通信的消息内容以 .aidl 后缀格式的文件默认保存在开发目录 /src/main/aidl 下。在 .adil 格式的文件中,编码规则与Java语言一致,其中内容与Java语言中的接口定义类似。

其中支持通信的消息类型分为两类,一类是基本类型,使用基本类型时不需要在文件开头增加导入包名的声明,这些基本类型包括Java语言中的八种基本数据类型, String, CharSequence, List, Map;还有一类是附加类型,除了第一类基本类型,其他在 .java 格式的文件中定义的任何实现android.os.Parcelable接口的类都可以作为附加类型消息传递。附加类型在使用时需要在 .aidl 文件开头增加导入该类型所在包名的声明。

关于附加类型的定义,以com.java.process.myinterface.Person类型为例。

不仅在 /src/main/java/com/java/process/myinterface 目录下创建 Person.java ,在其中定义Person类并实现Parcelable接口的相关方法。

还要在 /src/main/aidl/com/java/process/myinterface 目录下创建对应的 Person.aidl 文件,在其中除了指定包名外,还要声明该类为parcelable接口,示例代码如下

package com.java.process.myinterface;
parcelable Person;

.aidl.java 格式的文件中定义的接口相比有两处不同,其一是无需访问修饰符,其二是对基本数据类型增加数据流向标识符(包括数据流入接收方的in,数据在接收方修改后可流出的out, 数据可流向接收方并流出的inout )。

关于通信消息的定义,以com.java.process.myinterface.DataInterface类型为例。

只需要在 /src/main/aidl/com/java/process/myinterface 目录下创建 DataInterface.aidl 文件,在其中定义相关接口即可。示例代码如下

package com.java.process.myinterface;
import com.java.process.myinterface.Person;
interface DataInterface{
void setVersion(int version);
int getVersion();
void updatePerson(in Person person);
Person getMainPerson();
}

在通信消息定义之后,可以使用AndroidStudio的编译指令 Build - Make Project 编译当前项目,以使得AndroidStudio自动生成消息定义的 .java 文件。上文 DataInterface.aidl 文件在项目编译之后,会在 /build/generated/aidl_source_output_dir/debug/out/com/java/process/myinterface 目录下生成不可编辑的 DataInterface.java 文件,之后即可在项目中通过导包import com.java.process.myinterface.DataInterface;的形式正常使用该接口类了。

通信接收方的服务接收处理

通信接收方就是常说的服务端,通常是自定义的服务Service类,主要负责实现上述通信消息内容的接口,以此接收消息内容并处理。

针对自定义的服务Service类,这里就用到在加载服务文章中提到的绑定服务的生命周期了。重写onBind(Intent intent)方法,在该方法中返回android.os.IBinder类型的对象实例。

而这个IBinder对象是怎么创建的呢?在上文通信消息定义之后自动生成的接口类中,也自动生成了其Stub内部类,在创建该内部类的无参构造方法时,即可自动实现其相关接口方法,继续使用上文示例,这里在服务Service中的接口实现代码如下

package com.java.process.myinterface;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.Process;
import android.os.RemoteException;
import android.support.annotation.Nullable; import com.java.process.myinterface.DataInterface; public class ProcessService extends Service { private String versionName;
private Person person;
private final DataInterface.Stub binder=new DataInterface.Stub() { @Override
public String getVersionName() throws RemoteException {
//这里可以返回消息versionName给客户端
return versionName;
} @Override
public void setVersionName(String versionName) throws RemoteException {
//这里接收处理来自客户端的消息versionName
ProcessService.this.versionName=versionName;
} @Override
public void updatePerson(Person person) throws RemoteException {
//这里接收处理来自客户端的消息person
ProcessService.this.person.setName(person.getName());
ProcessService.this.person.setAge(person.getAge());
} @Override
public Person getMainPerson() throws RemoteException {
//这里可以返回消息person给客户端
return ProcessService.this.person;
}
}; @Override
public void onCreate() {
super.onCreate();
versionName="defaultName";
person=new Person();
person.setName("initName");
person.setAge(10);
} @Nullable
@Override
public IBinder onBind(Intent intent) {
return binder;
}
}

通信发送方的服务绑定

通信发送方就是所谓的客户端了,发送方是与上文中的接收方不在同一个进程的,所以发送方通常需要先绑定接收方的自定义服务Service,也就是在发送方通过上下文环境Context对象调用bindService(Intent service, ServiceConnection conn, int flags)方法,这里的参数在加载服务中已有详细说明。

conn 参数的onServiceConnected(ComponentName name, IBinder service)方法中通过调用静态方法(通信消息定义接口的内部类Stub.asInterface(IBinder service)方法)得到通信消息的定义接口的实例化对象。在当前客户端绑定自定义服务Service成功之后,回调该方法,即可将通信消息的定义接口的实例化对象赋值给全局变量使用。

conn 参数的onServiceDisconnected(ComponentName name)方法中,要记得将全局通信消息定义接口变量的实例化对象置为空,否则在当前客户端与自定义服务Service断开连接后,还调用全局变量的通信消息定义接口的实例化对象的相关方法,可能会出现OOM内存泄露的问题。

最后在需要发送消息的位置,只需要调用全局通信消息定义接口变量的相关方法即可。其中用到的示例代码如下

    private DataInterface dataInterface;
private ServiceConnection conn=new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
dataInterface = DataInterface.Stub.asInterface(service);
} @Override
public void onServiceDisconnected(ComponentName name) {
dataInterface = null;
} //此处省略对文本编辑控件editText的定义
//该方法在点击editText控件时回调
public void clickName(View view) {
try {
String setName = editText.getText().toString();
dataInterface.setVersionName(setName);
} catch (RemoteException e) {
e.printStackTrace();
}
}
};

Android系统编程入门系列之服务Service中的进程间通信的更多相关文章

  1. Android系统编程入门系列之服务Service齐头并进多线程任务

    在上篇文章中初步了解了Android系统的四大组件之一的服务Service,在服务内可以执行无用户交互的耗时操作任务,但是包括之前关于界面系列文章在内,生命周期方法都是在主线程内被系统回调的.如果直接 ...

  2. Android系统编程入门系列之广播接收者BroadcastReceiver实现进程间通信

    在前边几篇关于Android系统两个重要组件的介绍中,界面Activity负责应用程序与用户的交互,服务Service负责应用程序内部线程间的交互或两个应用程序进程之间的数据交互.看上去这两大组件就能 ...

  3. Android系统编程入门系列之加载界面Activity

    上回说到应用初始化加载及其生命周期,在Android系统调用Applicaiton.onCreate()之后,继续创建并加载清单文件中注册的首个界面即主Activity,也可称之为入口界面.主Acti ...

  4. Android系统编程入门系列之加载服务Service

    之前几篇文章简单梳理了在Android系统的四大组件之一,最主要的界面Activity中,使应用程序与用户进行交互响应的相关知识点,那对于应用程序中不需要与用户交互的逻辑,又要用到哪些内容呢?本文开始 ...

  5. Android系统编程入门系列之硬件交互——通信硬件Bluetooth

    通信硬件NFC的文章,虽然可以在Android系统中通过非直接接触的形式与支持NFC硬件的设备通信,但是也只能交互一些简短的标签内容,对大量的持续性数据,却并不能很好的支持.因此针对这个弊端,可以考虑 ...

  6. Android系统编程入门系列之应用环境及开发环境介绍

        作为移动端操作系统,目前最新的Android 11.0已经发展的比较完善了,现在也到了系统的整理一番的时间,接下来的系列文章将以Android开发者为中心,争取用归纳总结的态度对初级入门者所应 ...

  7. Android系统编程入门系列之应用间数据共享ContentProvider

    内容提供者ContentProvider与前文的界面Activity.服务Service.广播接收者BroadcastReveiver,并列称为Android的四大组件,均是需要自定义子类继承上述组件 ...

  8. Android系统编程入门系列之清单文件

    在上一篇文章中已经提到,Android系统加载应用程序之后,首先会读取该应用程序的AndroidManifest.xml清单文件,之后根据该清单文件加载后边的东西.所以要开发应用程序,自然要先知道清单 ...

  9. Android系统编程入门系列之应用内键值对数据的简单保存

    在应用程序间及与用户的通信交互过程中,会产生并传递一系列数据.针对这些数据,有部分是只在应用程序中使用的缓存数据,还有一部分是在不同位置多次或长时间使用的持久化数据. 对于缓存数据来说,通常以代码中定 ...

随机推荐

  1. Androidmanifest.xml文件格式详解(转载)

    https://www.jianshu.com/p/eaaae96473f6 来自简书大佬的

  2. php漏洞 md5函数漏洞

    0x01: 背景:php在处理哈希值时,用!=和==来比较的时候,如果哈希字符串以0E开头的时候,哈希值会默认为0,所以两个不同的字符串经过md5加密成哈希值,如果哈希值开头是0E的话,会默认成相等. ...

  3. cut和grep 选取命令

    cut命令 cut:将一段信息的某一段"切"出来,处理的信息是以行为单位.参数: -d :后接分隔字符,与-f一起使用: -f :依据-d的分隔字符将一段信息切割成为数段,用-f取 ...

  4. 基于YARP实现的FastGithub

    前言 最近开源的两个项目,先是FastGithub,旨在解决访问github抽风的问题.然后开发HttpMouse项目,基于yarp的http公网反向代理到内网的服务端与客户端库,在开发HttpMou ...

  5. C语言:位运算符总结

    位运算符:1.指对操作数以二进制位( bit)为单位进行的数据处理2.每一个二进制位只存放0或13. 取反:~  按位反 ~ 0变1 1变0 ~1=0 ~0=14.异或: ^ 相同为0,不相同为1 1 ...

  6. sql-5-事务,索引

    事务 1.ACID概念 原子性(Atomicity) 要么都成功,要么都失败 一致性(consistency) 事务前后的数据完整性保持一致 持久性(Durability) 事务一旦提交则不可逆,持久 ...

  7. PHP 多进程下载必应壁纸

    手里拿着锤子,看什么都像是钉子 在放假的这几天,断断续续的看了老李关于 PHP 多进程的文章. PHP多进程初探 --- 开篇 PHP多进程初探 --- 孤儿和僵尸 PHP多进程初探 --- 信号 P ...

  8. Beego和Vue的前后端分离跨域问题处理

    VUE封装的请求头(注意请求头,跨域要用到) 路径 utils/mereq.js import request from '@/utils/request' import qs from 'qs' e ...

  9. odoo里的rpc用法

    import odoorpcdb_name = 'test-12'user_name = 'admin'password = 'admin'# Prepare the connection to th ...

  10. win10家庭版无法访问samba

    1:本教程只针对win10家庭版用户,右键点击电脑选择属性就能看到自己的版本.(因为win10家庭版没有本地策略组) 2:针对一下连接不上的情况               3:连接不上的情况 ① : ...