iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >怎么在Android中使用AIDL实现进程间通信
  • 711
分享到

怎么在Android中使用AIDL实现进程间通信

androidaidl 2023-05-30 21:05:37 711人浏览 独家记忆
摘要

怎么在Android中使用aiDL实现进程间通信?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。一、概述AIDL 意思即 Android Interface Definiti

怎么在Android中使用aiDL实现进程间通信?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。

一、概述

AIDL 意思即 Android Interface Definition Language,翻译过来就是Android接口定义语言,是用于定义服务器和客户端通信接口的一种描述语言,可以拿来生成用于IPC的代码。从某种意义上说AIDL其实是一个模板,因为在使用过程中,实际起作用的并不是AIDL文件,而是据此而生成的一个IInterface的实例代码,AIDL其实是为了避免我们重复编写代码而出现的一个模板

设计AIDL这门语言的目的就是为了实现进程间通信。在Android系统中,每个进程都运行在一块独立的内存中,在其中完成自己的各项活动,与其他进程都分隔开来。可是有时候我们又有应用间进行互动的需求,比较传递数据或者任务委托等,AIDL就是为了满足这种需求而诞生的。通过AIDL,可以在一个进程中获取另一个进程的数据和调用其暴露出来的方法,从而满足进程间通信的需求

通常,暴露方法给其他应用进行调用的应用称为服务端,调用其他应用的方法的应用称为客户端,客户端通过绑定服务端的Service来进行交互

二、语法

AIDL的语法十分简单,与Java语言基本保持一致,需要记住的规则有以下几点:

AIDL文件以 .aidl 为后缀名

AIDL支持的数据类型分为如下几种:

  • 八种基本数据类型:byte、char、short、int、long、float、double、boolean

  • String,CharSequence

  • 实现了Parcelable接口的数据类型

  • List 类型。List承载的数据必须是AIDL支持的类型,或者是其它声明的AIDL对象

  • Map类型。Map承载的数据必须是AIDL支持的类型,或者是其它声明的AIDL对象

AIDL文件可以分为两类。一类用来声明实现了Parcelable接口的数据类型,以供其他AIDL文件使用那些非默认支持的数据类型。还有一类是用来定义接口方法,声明要暴露哪些接口给客户端调用,定向Tag就是用来标注这些方法的参数值

定向Tag。定向Tag表示在跨进程通信中数据的流向,用于标注方法的参数值,分为 in、out、inout 三种。其中 in 表示数据只能由客户端流向服务端, out 表示数据只能由服务端流向客户端,而 inout 则表示数据可在服务端与客户端之间双向流通。此外,如果AIDL方法接口的参数值类型是:基本数据类型、String、CharSequence或者其他AIDL文件定义的方法接口,那么这些参数值的定向 Tag 默认是且只能是 in,所以除了这些类型外,其他参数值都需要明确标注使用哪种定向Tag。定向Tag具体的使用差别后边会有介绍

明确导包。在AIDL文件中需要明确标明引用到的数据类型所在的包名,即使两个文件处在同个包名下

三、服务端编码

这里来实际完成一个例子作为示范,需要实现的功能是:客户端通过绑定服务端的Service的方式来调用服务端的方法,获取服务端的书籍列表并向其添加书籍,实现应用间的数据共享

首先是服务端的代码

新建一个工程,包名就定义为 com.czy.server

首先,在应用中需要用到一个 Book 类,而 Book 类是两个应用间都需要使用到的,所以也需要在AIDL文件中声明Book类,为了避免出现类名重复导致无法创建文件的错误,这里需要先建立 Book AIDL 文件,之后再创建 Book 类

右键点击新建一个AIDL文件,命名为 Book

怎么在Android中使用AIDL实现进程间通信

创建完成后,系统就会默认创建一个 aidl 文件夹,文件夹下的目录结构即是工程的包名,Book.aidi 文件就在其中

Book.aidl 文件中会有一个默认方法,可以删除掉

怎么在Android中使用AIDL实现进程间通信

此时就可以来定义Book类了,Book类只包含一个 Name 属性,并使之实现 Parcelable 接口

public class Book implements Parcelable { private String name; public Book(String name) {  this.name = name; } public String getName() {  return name; } public void setName(String name) {  this.name = name; } @Override public String toString() {  return "book name:" + name; } @Override public int describeContents() {  return 0; } @Override public void writeToParcel(Parcel dest, int flags) {  dest.writeString(this.name); } public void readFromParcel(Parcel dest) {  name = dest.readString(); } protected Book(Parcel in) {  this.name = in.readString(); } public static final Creator<Book> CREATOR = new Creator<Book>() {  @Override  public Book createFromParcel(Parcel source) {   return new Book(source);  }  @Override  public Book[] newArray(int size) {   return new Book[size];  } };}

现在再来修改 Book.aidl 文件,将之改为声明Parcelable数据类型的AIDL文件

package com.czy.server;parcelable Book;

此外,根据一开始的设想,服务端需要暴露给客户端一个获取书籍列表以及一个添加书籍的方法,这两个方法首先要定义在AIDL文件中,命名为 BookController.aidl,注意这里需要明确导包

package com.czy.server;import com.czy.server.Book;interface BookController { List<Book> getBookList(); void addBookInOut(inout Book book);}

上面说过,在进程间通信中真正起作用的并不是 AIDL 文件,而是系统据此而生成的文件,可以在以下目录中查看系统生成的文件。之后需要使用到当中的内部静态抽象类 Stub

创建或修改过AIDL文件后需要clean下工程,使系统及时生成我们需要的文件

怎么在Android中使用AIDL实现进程间通信

现在需要来创建一个 Service 供客户端远程绑定了,这里命名为 AIDLService

public class AIDLService extends Service { private final String TAG = "Server"; private List<Book> bookList; public AIDLService() { } @Override public void onCreate() {  super.onCreate();  bookList = new ArrayList<>();  initData(); } private void initData() {  Book book1 = new Book("活着");  Book book2 = new Book("或者");  Book book3 = new Book("叶应是叶");  Book book4 = new Book("https://GitHub.com/leavesC");  Book book5 = new Book("Http://www.jianshu.com/u/9df45b87cfdf");  Book book6 = new Book("http://blog.csdn.net/new_one_object");  bookList.add(book1);  bookList.add(book2);  bookList.add(book3);  bookList.add(book4);  bookList.add(book5);  bookList.add(book6); } private final BookController.Stub stub = new BookController.Stub() {  @Override  public List<Book> getBookList() throws RemoteException {   return bookList;  }  @Override  public void addBookInOut(Book book) throws RemoteException {   if (book != null) {    book.setName("服务器改了新书的名字 InOut");    bookList.add(book);   } else {    Log.e(TAG, "接收到了一个空对象 InOut");   }  } }; @Override public IBinder onBind(Intent intent) {  return stub; }}

可以看到, onBind 方法返回的就是 BookController.Stub 对象,实现当中定义的两个方法

最后,服务端还有一个地方需要注意,因为服务端的Service需要被客户端来远程绑定,所以客户端要能够找到这个Service,可以通过先指定包名,之后再配置Action值或者直接指定Service类名的方式来绑定Service

如果是通过指定Action值的方式来绑定Service,那还需要将Service的声明改为如下所示:

怎么在Android中使用AIDL实现进程间通信

本例子采用配置 Action 的方案

四、客户端编码

客户端需要再创建一个新的工程,包名命名为 com.czy.client

首先,需要把服务端的AIDL文件以及Book类复制过来,将 aidl 文件夹整个复制到和Java文件夹同个层级下,不需要改动任何代码

怎么在Android中使用AIDL实现进程间通信

之后,需要创建和服务端Book类所在的相同包名来存放 Book类

怎么在Android中使用AIDL实现进程间通信

修改布局文件,添加两个按钮

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:gravity="center" android:orientation="vertical"> <Button  android:id="@+id/btn_getBookList"  android:layout_width="match_parent"  android:layout_height="wrap_content"  android:text="获取书籍列表" /> <Button  android:id="@+id/btn_addBook_inOut"  android:layout_width="match_parent"  android:layout_height="wrap_content"  android:text="InOut 添加书籍" /></LinearLayout>
public class MainActivity extends AppCompatActivity { private final String TAG = "Client"; private BookController bookController; private boolean connected; private List<Book> bookList; private ServiceConnection serviceConnection = new ServiceConnection() {  @Override  public void onServiceConnected(ComponentName name, IBinder service) {   bookController = BookController.Stub.asInterface(service);   connected = true;  }  @Override  public void onServiceDisconnected(ComponentName name) {   connected = false;  } }; private View.OnClickListener clickListener = new View.OnClickListener() {  @Override  public void onClick(View v) {   switch (v.getId()) {    case R.id.btn_getBookList:     if (connected) {      try {       bookList = bookController.getBookList();      } catch (RemoteException e) {       e.printStackTrace();      }      log();     }     break;    case R.id.btn_addBook_inOut:     if (connected) {      Book book = new Book("这是一本新书 InOut");      try {       bookController.addBookInOut(book);       Log.e(TAG, "向服务器以InOut方式添加了一本新书");       Log.e(TAG, "新书名:" + book.getName());      } catch (RemoteException e) {       e.printStackTrace();      }     }     break;   }  } }; @Override protected void onCreate(Bundle savedInstanceState) {  super.onCreate(savedInstanceState);  setContentView(R.layout.activity_main);  findViewById(R.id.btn_getBookList).setOnClickListener(clickListener);  findViewById(R.id.btn_addBook_inOut).setOnClickListener(clickListener);  bindService(); } @Override protected void onDestroy() {  super.onDestroy();  if (connected) {   unbindService(serviceConnection);  } } private void bindService() {  Intent intent = new Intent();  intent.setPackage("com.czy.server");  intent.setAction("com.czy.server.action");  bindService(intent, serviceConnection, Context.BIND_AUTO_CREATE); } private void log() {  for (Book book : bookList) {   Log.e(TAG, book.toString());  } }}

两个按钮分别用于获取服务端的书籍列表和添加书籍,在添加书籍时,服务端还改变了Book对象的Name属性,据此观察客户端和服务端数据的变化情况

首先点击获取书籍列表,数据获取无误

怎么在Android中使用AIDL实现进程间通信

再点击按钮添加书籍,可以看到,服务端对数据的修改也同时同步到了客户端这边

怎么在Android中使用AIDL实现进程间通信

到此为止,客户端和服务端之间的通信已经实现了,客户端获取到了服务端的数据,也向服务端传送了数据

五、定向Tag

最后,我再来讲下三种定向Tag之间的差别。上边使用的是 InOut 类型,服务端对数据的改变同时也同步到了客户端,因此可以说两者之间数据是双向流动的

In 类型的表现形式是:数据只能由客户端传向服务端,服务端对数据的修改不会影响到客户端

Out类型的表现形式是:数据只能由服务端传向服务端,即使客户端向方法接口传入了一个对象,该对象中的属性值也是为空的,即不包含任何数据,服务端获取到该对象后,对该对象的任何操作,就会同步到客户端这边

这里再来实际演示一下

先修改服务器端的 BookController.aidl 文件,向之添加两个新方法

package com.czy.server;import com.czy.server.Book;interface BookController { List<Book> getBookList(); void addBookInOut(inout Book book); void addBookIn(in Book book); void addBookOut(out Book book);}

则 AIDLService 类的 BookController.Stub 对象就需要修改为如下所示:

private final BookController.Stub stub = new BookController.Stub() {  @Override  public List<Book> getBookList() throws RemoteException {   return bookList;  }  @Override  public void addBookInOut(Book book) throws RemoteException {   if (book != null) {    book.setName("服务器改了新书的名字 InOut");    bookList.add(book);   } else {    Log.e(TAG, "接收到了一个空对象 InOut");   }  }  @Override  public void addBookIn(Book book) throws RemoteException {   if (book != null) {    book.setName("服务器改了新书的名字 In");    bookList.add(book);   } else {    Log.e(TAG, "接收到了一个空对象 In");   }  }  @Override  public void addBookOut(Book book) throws RemoteException {   if (book != null) {    Log.e(TAG, "客户端传来的书的名字:" + book.getName());    book.setName("服务器改了新书的名字 Out");    bookList.add(book);   } else {    Log.e(TAG, "接收到了一个空对象 Out");   }  } };

同步修改客户端的 BookController.aidl 文件

向布局文件多增添两个按钮,分别用于添加不同的 定向Tag 的数据

private View.OnClickListener clickListener = new View.OnClickListener() {  @Override  public void onClick(View v) {   switch (v.getId()) {    case R.id.btn_getBookList:     if (connected) {      try {       bookList = bookController.getBookList();      } catch (RemoteException e) {       e.printStackTrace();      }      log();     }     break;    case R.id.btn_addBook_inOut:     if (connected) {      Book book = new Book("这是一本新书 InOut");      try {       bookController.addBookInOut(book);       Log.e(TAG, "向服务器以InOut方式添加了一本新书");       Log.e(TAG, "新书名:" + book.getName());      } catch (RemoteException e) {       e.printStackTrace();      }     }     break;    case R.id.btn_addBook_in:     if (connected) {      Book book = new Book("这是一本新书 In");      try {       bookController.addBookIn(book);       Log.e(TAG, "向服务器以In方式添加了一本新书");       Log.e(TAG, "新书名:" + book.getName());      } catch (RemoteException e) {       e.printStackTrace();      }     }     break;    case R.id.btn_addBook_out:     if (connected) {      Book book = new Book("这是一本新书 Out");      try {       bookController.addBookOut(book);       Log.e(TAG, "向服务器以Out方式添加了一本新书");       Log.e(TAG, "新书名:" + book.getName());      } catch (RemoteException e) {       e.printStackTrace();      }     }     break;   }  } };

此外,还有个地方需要修改下,即Book类。因为 Out 类型的确会使得客户端传一个不包含任何数据的对象回服务端,但该对象却不是直接就等于 null ,所以说明系统还是需要实例化 Book 类,但当前 Book 类只有一个有参构造函数,所以还需要修改 Book 类,为之添加一个无参构造函数以供系统使用

分别点击三个按钮,可以看到,服务端在获取到客户端以Out方式传来的Book对象时,的确是不包含书名这个属性值

怎么在Android中使用AIDL实现进程间通信

关于怎么在Android中使用AIDL实现进程间通信问题的解答就分享到这里了,希望以上内容可以对大家有一定的帮助,如果你还有很多疑惑没有解开,可以关注编程网精选频道了解更多相关知识。

--结束END--

本文标题: 怎么在Android中使用AIDL实现进程间通信

本文链接: https://www.lsjlt.com/news/221423.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 怎么在Android中使用AIDL实现进程间通信
    怎么在Android中使用AIDL实现进程间通信?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。一、概述AIDL 意思即 Android Interface Definiti...
    99+
    2023-05-30
    android aidl
  • Android中怎么实现进程间通信
    这篇文章给大家介绍Android中怎么实现进程间通信,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。由于android系统中应用程序之间不能共享内存。因此,在不同应用程序之间交互数据(跨进程通讯)就稍微麻烦一些。进程间通...
    99+
    2023-06-04
  • Android AIDL实现跨进程通信的示例代码
    AIDL是Android接口定义语言,它可以用于让某个Service与多个应用程序组件之间进行跨进程通信,从而可以实现多个应用程序共享同一个Service的功能。实现步骤例:用 A程序去访问 B程序的MyService.java服务 在B...
    99+
    2023-05-30
    android 跨进程通信 aidl
  • Android进程间使用Intent进行通信
    安卓使用Intent来封装程序的“调用意图”,使用Intent可以让程序看起来更规范,更易于维护。 除此之外,使用Intent还有一个好处:有些时候我们只是想...
    99+
    2023-02-28
    Android Intent通信 Android进程通信
  • Android Messenger实现进程间双向通信
    简介 Messenger是安卓进程间通信 (IPC) 最为简单的方式,可以实现进程间双向通信。详见官网介绍 代码实现 服务端应用实现 MessengerService接收客户端发送的...
    99+
    2024-04-02
  • Node中的进程间通信怎么实现
    这篇“Node中的进程间通信怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Node...
    99+
    2024-04-02
  • golang进程间通信怎么实现
    在Go语言中,有多种方式可以实现进程间通信。以下是一些常见的方法: 使用管道(Pipe):管道是进程间通信的一种简单而有效的方式...
    99+
    2023-10-25
    golang
  • Linux进程间通信怎么实现
    这篇文章主要讲解了“Linux进程间通信怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Linux进程间通信怎么实现”吧!共享内存共享内存可以说是最有用的进程间通信方式,也是最快的IP...
    99+
    2023-07-05
  • Android进程间如何使用Intent进行通信
    这篇文章主要介绍“Android进程间如何使用Intent进行通信”,在日常操作中,相信很多人在Android进程间如何使用Intent进行通信问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Android进程...
    99+
    2023-07-05
  • 怎么在java中实现线程间通信
    这篇文章将为大家详细讲解有关怎么在java中实现线程间通信,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。线程间通信:由于多线程共享地址空间和数据空间,所以多个线程间的通信是一个线程的数据可以...
    99+
    2023-05-30
    java
  • Android Messenger实现进程间通信及其原理
    前言 之前分析Android消息机制的源码时,曾遇到过replyTo、IMessenger等属性字段,当时只是说这些字段用于进程间通信,并未作深入分析。今天这篇文字就来演示一下使用M...
    99+
    2024-04-02
  • C++中如何使用管道实现进程间通信
    在C++中使用管道实现进程间通信可以通过以下步骤: 包含相关的头文件: #include <iostream> #i...
    99+
    2024-04-02
  • C++中怎么用共享文件实现进程间通信
    在C++中使用共享文件实现进程间通信,可以通过以下步骤实现: 创建一个共享文件,可以使用open()函数或者fopen()函数创建...
    99+
    2024-04-02
  • Electron进程间通信的实现
    目录主进程与渲染进程之间通信ipc模块 + window.webContentsremote模块渲染进程之间通信使用Electron开发出来的桌面应用都是多进程的,其中包含了一个主进...
    99+
    2024-04-02
  • android线程间怎么通信
    在Android中,线程间通信可以通过以下几种方式实现:1. Handler:可以通过Handler对象在不同的线程之间发送消息。每...
    99+
    2023-08-19
    android
  • Android怎么使用ContentProvider实现跨进程通讯
    这篇“Android怎么使用ContentProvider实现跨进程通讯”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“And...
    99+
    2023-07-05
  • Android怎么使用GRPC进行通信
    这篇文章主要介绍“Android怎么使用GRPC进行通信”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Android怎么使用GRPC进行通信”文章能帮助大家解决问题。引言Android作为一个开发平...
    99+
    2023-07-05
  • Android中的多进程通信怎么利用继承Binder类实现
    Android中的多进程通信怎么利用继承Binder类实现?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。服务端代码,BinderService.java:首先继...
    99+
    2023-05-31
    android 多进程 binder
  • 怎么在Android中实现Socket通信传输
    今天就跟大家聊聊有关怎么在Android中实现Socket通信传输,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。Socket本质上就是Java封装了传输层上的TCP协议(注:UDP用...
    99+
    2023-05-30
    android socket
  • Electron进程间通信如何实现
    今天小编给大家分享一下Electron进程间通信如何实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。使用Electron开...
    99+
    2023-06-30
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作