在这篇文章中,我将描述如何在Android中使用远程服务。 此类服务是其他进程可以使用RPC(远程过程调用)使用的服务。 在另一篇文章中,我们谈论本地服务,换句话说,托管服务的应用程序可以使用它。 当我们想要创建一些新功能并将其作为库分发时,AIDL服务非常有用。 一个有趣的方面,我们在开发AIDL服务时应该考虑的是,它可以由运行在不同进程中的组件调用/使用。 为了支持IPC(进程间通信),我们必须在Android中定义一个接口,该接口描述将向客户端公开的方法。 要创建此界面,我们使用AIDL(Android界面定义语言)。

考虑到这些远程服务可以作为库分发,我们必须选择要提供给客户端的内容(如jar),以便它可以调用和使用我们的服务。 那么,重要的是要拥有正确的项目结构,以便我们可以创建仅包含必需类的客户端jar。 在本文的其余部分中,我们也将重点放在这方面。
作为示例,我们将使用上次描述的获得股票报价的示例。

定义远程AIDL服务

为了创建AIDL服务,我们有:

  1. 使用AIDL定义和创建服务接口
  2. 实现我们的服务并重写onBind方法以返回我们的接口
  3. 定义客户端和服务器交换的对象,并在较低的OS级别上对其进行解构,以便可以将它们封送和取消封送。 换句话说,我们的类必须实现Parcelable接口。
  4. 在Manifest.xml文件中配置我们的服务

在我们的示例中,我们知道我们只想知道股票报价,因此为简单起见,我们的界面仅由一种称为getQuote的方法构成。 在这种方法中,我们传递了Stock pojo类,该类保存有关股票代码和将由我们的服务填充的值的信息。 我们的pojo课称为Stock。 因此,考虑到所有内容,我们认为AIDL为:

package com.survivingwithandroid.aidlservicetutorial.service;

import com.survivingwithandroid.aidlservicetutorial.service.Stock;

interface IStockService{
    void getQuote(Stock stock);
}

请注意,在第3行,我们仅导入Stock定义,在第6行,我们定义了方法。 另一方面,我们必须在AIDL中定义我们的Stock pojo:

package com.survivingwithandroid.aidlservicetutorial.service;

parcelable Stock;

这样,我们定义了服务接口。 如果使用Eclipse,则可以将这两个文件放在“源”和“包”名称下。 Eclipse将创建使用该服务所需的一切。

实施AIDL远程服务

现在我们有了界面,因此我们可以实现“真实的” Android服务:

public class StockService extends Service {
    ...
 
    @Override
    public IBinder onBind(Intent intent) {
        Log.d("Srv", "OnBind");
        final ResultReceiver rec = (ResultReceiver) intent.getParcelableExtra("rec");
         
        return new IStockService.Stub() {
            @Override
            public void getQuote(Stock stock) throws RemoteException {
                (new Thread(new Worker(stock, rec))).start();
            }
        };
    }
 
 
}

像往常一样在Android中创建服务,我们扩展了SDK提供的Service类(第1行)。 当我们要实现远程服务时,最重要的事情是重写onBind方法并返回接口实现(第7行)。 在第9行,我们实现了调用线程以获取股票报价的接口方法getQuote。 注意,这里我们使用ResultReceiver方法将结果通知给客户端

客户实施

最后一步是实现调用和使用服务的客户端。 要开发客户,我们需要:

  1. 服务接口(在AIDL中描述)
  2. 客户端和服务器交换的pojo类

通过这两个元素,我们可以创建我们的客户。 稍后我们将看到如何构建项目。

使用远程服务时,我们必须将客户端“绑定”到远程服务。 例如,我们可以在我们的Activity的onCreate方法中使用bindService方法来做到这一点:

Intent i = new Intent(IStockService.class.getName());
...
bindService(i, serviceConnection, Context.BIND_AUTO_CREATE);

在serviceConnection是侦听器的地方,我们提供了一些可用于监视服务连接状态的回调方法。 因此,我们必须创建一个ServiceConnection实例来处理:

  1. 服务连接事件
  2. 服务中断事件
ServiceConnection serviceConnection = new ServiceConnection() {
     @Override
     public void onServiceConnected(ComponentName name, IBinder binder) {
         Log.d("Srv", "Service connected!");
         service = IStockService.Stub.asInterface(binder);
         Log.d("Srv", "Service interface ["+service+"]");
     }

     @Override
     public void onServiceDisconnected(ComponentName name) {
         Log.d("Srv", "Service disconnected!");
         service = null;
     }
 };

在第5行,我们最终获得了服务接口,该接口可用于在服务端调用远程方法。 有了接口后,就可以像调用方法在我们的类之一中一样调用它。

项目结构

开发远程服务时,我们必须考虑的一个重要方面是客户需要调用我们的服务。 一种非常简单的方法是仅在一个库中混合pojo类,辅助和服务实现,并将其分发给想要使用我们的服务的开发人员。 即使这种方法可行,也有一些缺点:

  1. 该jar可能具有较大的尺寸,并且客户端应用程序开发人员必须将其包含在其应用程序分发中
  2. 我们将jar分发为我们的服务实现,也许将其保存在其他位置更明智
  3. 即使我们不修改服务接口和pojos,而仅修改服务实现,我们的客户端和服务器jar也不对齐

我认为,以正确的方式组织项目范围更广,以便我们可以仅将客户真正需要的类分发给客户开发人员。 如果使用Eclipse,则可以为服务器(我们的服务实现)创建两个不同的项目,为客户端lib创建一个。 我们要记住的重要一件事是将最后一个项目标记为库:

android远程控制 代码 android远程开发_android远程控制 代码

现在,在AIDLServiceLib项目中,我们添加了客户端所需的所有内容:

  1. AIDL定义
  2. AIDL中的Pojos参考

android远程控制 代码 android远程开发_java_02

而在AIDLServiceTutorial中,我们有:

android远程控制 代码 android远程开发_android远程控制 代码_03

现在,只需在这两个项目之间进行引用即可,换句话说,AIDLServiceTutorial使用AIDLServiceLib作为库:

android远程控制 代码 android远程开发_java_04

现在,如果我们要创建一个客户,我们只需创建另一个项目:

android远程控制 代码 android远程开发_spring_05

通过这种方式,我们将客户端库类与服务实现解耦,并且可以分发与AIDLServiceLib相关的jar。 最后要记住的是将AIDLServiceLib设置为AIDLServiceClient的lib。

  • 源代码即将发布!

翻译自: https://www.javacodegeeks.com/2014/04/android-remote-service-tutorialaidl-how-to-structure-the-project-and-lib.html