在这篇文章中,我将描述如何在Android中使用远程服务。 此类服务是其他进程可以使用RPC(远程过程调用)使用的服务。 在另一篇文章中,我们谈论本地服务,换句话说,托管服务的应用程序可以使用它。 当我们想要创建一些新功能并将其作为库分发时,AIDL服务非常有用。 一个有趣的方面,我们在开发AIDL服务时应该考虑的是,它可以由运行在不同进程中的组件调用/使用。 为了支持IPC(进程间通信),我们必须在Android中定义一个接口,该接口描述将向客户端公开的方法。 要创建此界面,我们使用AIDL(Android界面定义语言)。
考虑到这些远程服务可以作为库分发,我们必须选择要提供给客户端的内容(如jar),以便它可以调用和使用我们的服务。 那么,重要的是要拥有正确的项目结构,以便我们可以创建仅包含必需类的客户端jar。 在本文的其余部分中,我们也将重点放在这方面。
作为示例,我们将使用上次描述的获得股票报价的示例。
定义远程AIDL服务
为了创建AIDL服务,我们有:
- 使用AIDL定义和创建服务接口
- 实现我们的服务并重写onBind方法以返回我们的接口
- 定义客户端和服务器交换的对象,并在较低的OS级别上对其进行解构,以便可以将它们封送和取消封送。 换句话说,我们的类必须实现Parcelable接口。
- 在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方法将结果通知给客户端 。
客户实施
最后一步是实现调用和使用服务的客户端。 要开发客户,我们需要:
- 服务接口(在AIDL中描述)
- 客户端和服务器交换的pojo类
通过这两个元素,我们可以创建我们的客户。 稍后我们将看到如何构建项目。
使用远程服务时,我们必须将客户端“绑定”到远程服务。 例如,我们可以在我们的Activity的onCreate方法中使用bindService方法来做到这一点:
Intent i = new Intent(IStockService.class.getName());
...
bindService(i, serviceConnection, Context.BIND_AUTO_CREATE);
在serviceConnection是侦听器的地方,我们提供了一些可用于监视服务连接状态的回调方法。 因此,我们必须创建一个ServiceConnection实例来处理:
- 服务连接事件
- 服务中断事件
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类,辅助和服务实现,并将其分发给想要使用我们的服务的开发人员。 即使这种方法可行,也有一些缺点:
- 该jar可能具有较大的尺寸,并且客户端应用程序开发人员必须将其包含在其应用程序分发中
- 我们将jar分发为我们的服务实现,也许将其保存在其他位置更明智
- 即使我们不修改服务接口和pojos,而仅修改服务实现,我们的客户端和服务器jar也不对齐
我认为,以正确的方式组织项目范围更广,以便我们可以仅将客户真正需要的类分发给客户开发人员。 如果使用Eclipse,则可以为服务器(我们的服务实现)创建两个不同的项目,为客户端lib创建一个。 我们要记住的重要一件事是将最后一个项目标记为库:
现在,在AIDLServiceLib项目中,我们添加了客户端所需的所有内容:
- AIDL定义
- AIDL中的Pojos参考
而在AIDLServiceTutorial中,我们有:
现在,只需在这两个项目之间进行引用即可,换句话说,AIDLServiceTutorial使用AIDLServiceLib作为库:
现在,如果我们要创建一个客户,我们只需创建另一个项目:
通过这种方式,我们将客户端库类与服务实现解耦,并且可以分发与AIDLServiceLib相关的jar。 最后要记住的是将AIDLServiceLib设置为AIDLServiceClient的lib。
- 源代码即将发布!