六.AIDL实现跨进程通讯

前言

在Android开发过程中,极少会用到跨进程通讯,但是一旦遇到跨进程通讯的时候,便会牵涉到服务的知识。那么,今天就让我们来学习下如何利用服务与AIDL的知识来实现跨进程通讯。

今天涉及到的知识点:

  1. 创建 aidl 文件
  2. 创建 A端供调用的 服务 ----- TestDemo项目
  3. 创建 B 端项目 ---- ATestDemo,用于跨进程调用 A端项目中的方法
  4. 需要注意的点
  5. 效果图与项目结构图
  6. A端项目中服务源码 及 B端项目中跨进程调用源码
  7. 参考资料

先来波效果图


1.gif

前提

本文是以 A端----TestDemo 项目创建可被调用方法,然后在 B端----ATestDemo 项目中去调用 A端 中的方法进行讲解 aidl 跨进程调用相关知识点的。其中A端与B端都是一个独立的项目。

一. 创建 aidl 文件

作为aidl跨进程调用,那么A端和B端都需要创建aidl文件,并且路径及文件名都要保持一致。那么,让我们先在A端----TestDemo中创建aidl。
新建 TestDemo项目,然后 File -----> New -----> AIDL -----> AIDL File,然后输入aidl文件名,我输入的是 Iplayer,新建完后,你的项目中将出现如下文件夹及文件


image.png

让我们打开 Iplayer.aidl 文件,可以看到以下代码:

// Iplayer.aidl
package com.example.testdemo;

interface Iplayer {

}

在里面添加我需要跨进程通讯的三个接口,则新的 Iplayer.aidl 文件内容如下:

// Iplayer.aidl
package com.example.testdemo;

interface Iplayer {

    void start();
    void pause();
    void stop();

}

这里需要注意的是,Iplayer.aidl 文件中的 interface 无 public 修饰,里面的方法(如:start(),pause()等)也没有public修饰,interface 名 Iplayer 必须与 Iplayer.aidl的文件名保持一致。
ok,这样的话,我们就将 A端----TestDemo 项目中的 aidl 创建完毕。

二.创建 A端供调用的 服务 ----- TestDemo项目

2.1 创建bind对象及服务Service

接着需要在 A端----TestDemo 项目中创建服务Service,然后在服务中提供一些供 B端调用的方法。
这里我新建了一个“绑定式”服务StuService,继承自Service,如下:

public class StuService extends Service {
    
}

然后在 StuService 中新建一个MediaPlayer,提供播放等方法。
接着在 StuService 中用 aidl相关方法创建一个 Bind 对象,bind的初始化为一个匿名对象,然后里面会出现 start,pause,stop三个方法,我们会在这三个方法中去调用MediaPlayer提供的几个播放,暂停和停止的方法,主要用于测试。
最后,在“绑定式”服务StuService的onBind方法中,返回aidl相关方法创建bind对象,如下:

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        LogUtil.i("==========启动服务===a==");
        return mBind;
    }
2.2 在A端----TestDemo 的manifast.xml中注册StuService

在A端----TestDemo 的manifast.xml中注册StuService如下:

  
        
            
                
            
        

    //其他代码省略
    //......
    

这里需要注意的是 service标签和action标签中的 name属性要保持唯一性(我在之前的文章已经提到过),然后还需要注意的是,因为涉及到跨进程通讯,所以 service标签 中还要添加以下属性:

 android:exported="true"

到这里,我们A端----TestDemo项目中的服务算是准备完毕了,然后看下A端----TestDemo中的MainActivity代码(这里MainActivity中无需做任何服务相关的处理):

public class MainActivity extends AppCompatActivity implements View.OnClickListener{

    private TextView mTv;
    private Button mBtn;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        initData();
        setListener();
    }

    private void initData(){
        mTv = findViewById(R.id.tv);
        mBtn = findViewById(R.id.btn);

    }

    private void setListener() {
        mBtn.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
            case R.id.btn:
                test();
                break;
            default:
                break;
        }
    }

    private void test() {

    }

    @Override
    protected void onDestroy() {
        super.onDestroy();

    }
}

至此,A端----TestDemo准备完毕。下面看看B端的准备

三.创建 B端 ---- ATestDemo项目,用于跨进程调用 A端项目中的方法

3.1 将 A端----TestDemo项目中的aidl文件夹复制到B 端 ---- ATestDemo项目中

先将A端----TestDemo调制Project模式(一般我设置的是Android项目模式,但是会导致aidl文件夹复制不出来),如下:


image.png

可以看到aidl文件夹在A端----TestDemo项目中的路径为:

Project -----> app -----> src -----> main -----> aidl

ok,然后复制这个aidl文件夹,打开 B端 ---- ATestDemo项目(此项目也以Project模式展示),按

Project -----> app -----> src -----> main

找到main文件夹,然后将 aidl文件夹复制进去,复制后,项目目录为:


image.png
3.2 在B端 ---- ATestDemo项目中调用 A端----TestDemo项目中的方法

aidl通讯,我们必须先在B端 ---- ATestDemo项目中获取aidl对象,即Iplayer对象,只有先获取Iplayer对象了,才能调用Iplayer中的方法。
在 A端----TestDemo项目中, Iplayer是作为一个“绑定式”服务中的bind对象存在,所以要获取Iplayer,我们需要在B端 ---- ATestDemo项目中通过启动 A端----TestDemo项目中 的服务 StuService来获取Iplayer对象。
ok,所以在 B端 ---- ATestDemo的MainActivity中,我们需要启动A端----TestDemo项目中 的服务 StuService。
这里需要通过创建服务对象,然后在对象的匿名方法中获取 Iplayer对象。
下面重点讲讲如何在B端中调用A端的服务StuService,先看看B端中绑定A端的服务StuService的代码:

        //绑定服务
        Intent intent = new Intent();
        intent.setAction("com.example.function.StuService");//设置远程服务的action
        intent.setPackage("com.example.testdemo");//设置服务端的package
        bindService(intent, connection, BIND_AUTO_CREATE);

需要注意的是,由于B项目中无法获取A项目中的类,所以intent不能通过显示调用,假如你在B项目中直接:

Intent intent = new Intent(MainActivity.this,StuService.class);

是会报以下错误的:

Service Intent must be explicit:

所以只能通过intent.setXXX系列方法来绑定 A端----TestDemo项目中的服务StuService,这里需要注意的是:

 intent.setAction("com.example.function.StuService");//设置远程服务的action

中setAction的值是 A端----TestDemo项目中服务StuService中注册是action,让我们再来回顾下A端----TestDemo项目中服务在manifast.xml中的注册代码:


image.png

然后

 intent.setPackage("com.example.testdemo");//设置服务端的package

中setPackage为A端----TestDemo项目的包名,让我们回顾下A端----TestDemo项目manifast.xml中的包名


image.png

一切准备就绪,在 B端 ---- ATestDemo项目中启动跨进程的StuService后,便会获得Iplayer对象,接着在要调用的地方,调用:

       //播放
        if(mIplayer!=null) {
            try {
                mIplayer.start();
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }else{
            Log.i("pei","====请先运行textdemo项目======");
        }

便会调用到A项目中的播放方法。

四.需要注意的点

B端 ---- ATestDemo项目 跨进程调用 A端----TestDemo项目中的方法需要注意以下几点:

  • B端调用A端方法是通过aidl接口,然后在A端建一个可跨进程绑定的服务来实现的
  • B端与A端是两个独立的项目
  • B端与A端需要共同运行在同意设备上
  • B端调用A端方法,需要先启动A项目且保证A端正常运行,再启动B项目。
    若B点击调用方法时,A端未启动,则B端调用不生效且无反馈。
    若B点击调用方法时,A端已关闭,则B端调用不生效且无反馈。
    若B先启动,A后启动,则B端调用不生效且无反馈。

五.效果图与项目结构图

效果图


1.gif

A端----TestDemo项目结构图


image.png

B端----ATestDemo项目结构图
image.png

六.A端项目中服务源码 及 B端项目中跨进程调用源码

A端----TestDemo项目中“绑定式”服务StuService代码如下:

你可能感兴趣的:(六.AIDL实现跨进程通讯)