android AIDL服务,Android开发社招面试经验

92.static final int TRANSACTION_getQuote = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);

93.}

94.public double getQuote(java.lang.String ticker) throws android.os.RemoteException;

95.}

对于所生成的类,请注意以下几点。

在 AIDL 文件中定义的接口在生成的代码中实现为接口 (也就是说,有一个名为IStockQuoteService的接口)。

名为Stub的 static final 抽象类扩展了 android.os.Binder并实现了 IStockQuoteService。请注意该类是一个抽象类。

名为 Proxy的内部类实现了 IStockQuoteService, 后者是Stub类的代理。

AIDL文件必须位于应该包含所生成文件的包中(在AIDL文件的包声明中指定)。

在服务类中实现AIDL接口

上边我们为股票报价服务定义了 AIDL 文件并生成了绑定文件。现在我们将提供该服务的实现。要实现服务的接口,需要编写一个类来扩展 android.app.Service 并实现 IStockQuoteService接口。我们将编写的类 命名为 StockQuoteService。为了将服务向客户端公开,StockQuoteService 需要提供onBind()方法 的实现,我们还需要将一些配置信息添加到 AndroidManifest.xml文件中。  下面给出 我们服务类得实现。

IStockQuoteService 服务实现。

package com.androidbook.stockquoteservice;

import android.app.Service;

import android.content.Intent;

import android.os.IBinder;

import android.os.RemoteException;

import android.util.Log;

public class StockQuoteService extends Service {

private static final String TAG = “StockQuoteService”;

public class StockQuoteServiceImpl extends IStockQuoteService.Stub {

@Override

public double getQuote(String ticker) throws RemoteException {

Log.v(TAG, "getQuote() called for " + ticker);

return 20.0;

}

}

@Override

public void onCreate() {

super.onCreate();

Log.v(TAG, “onCreate called”);

}

@Override

public void onDestroy() {

super.onDestroy();

Log.v(TAG, “onDestory() called”);

}

@Override

public void onStart(Intent intent, int startId) {

super.onStart(intent, startId);

Log.v(TAG, “onStart() called”);

}

@Override

public IBinder onBind(Intent intent) {

Log.v(TAG, “onBind() called”);

return new StockQuoteServiceImpl();

}

}

在这个服务类中大家可以看到我们实现了 onBind() 方法。从AIDL文件生成的 Stub类是抽象类并且它实现了 IStockQuoteService接口。在我们的服务实现中,有一个扩展了 Stub类得内部类,名为 StockQuoteServiceImpl。此类充当着远程服务实现,而且 onBind()方法会返回此类的实例。到此,我们有了一个有效的 ADIL服务,但是外部的客户端还无法连接到它。

要将服务向客户端公开,需要在AndroidManifest.xml文件中添加服务声明,而这一次我们需要一个Intent 过滤器来公开服务,如下。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

从客户端应用程序调用服务

当客户端与服务通信时,它们之间必须有一个协议或契约。在Android中,这个契约就是AIDL。所以,使用服务的第一步是,获取服务的 AIDL文件并将其复制到客户端项目中。当将AIDL文件复制到客户端项目时,AIDL 编译器将创建一个接口定义文件,这个文件与我们在服务端定义的文件相同。这会向客户端公开所有的方法、参数并返回服务的类型。我们创建一个新项目并复制AIDL文件。

(1)创建一个新的Android项目, 将其命名为 StockQuoteClient。使用不同的包名称比如 com.androidbook.stockquoteclient。在Create Activity字段中使用 MainActivity注意不要把IStockQuoteService.aidl文件放到这包中,这个包只有一个MainActivity 类。

(2)在此项目中新建一个包 com.androidbook.stockquoteservice,放在src目录下。

(3)将IStockQuoteService.aidl文件从 StockQuoteService 项目也就是我们服务端得项目复制到新建的包中。复制过来之后,AIDL编译器会自动生成关联的java文件。

重新生成的服务接口充当着客户端与服务之间的契约。下一步是获取服务的引用,以便调用getQuote()方法。对于远程服务,必须调用 bindService()方法,而不是 startService()方法。

客户端布局文件

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

android:orientation=“vertical”

android:layout_width=“fill_parent”

android:layout_height=“fill_parent”

<Button

android:id=“@+id/bindBtn”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:text=“Bind”/>

<Button

android:id=“@+id/callBtn”

android:layout_height=“wrap_content”

android:layout_width=“wrap_content”

android:text=“Call Again”/>

<Button

android:id=“@+id/unbindBtn”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:text=“UnBind”/>

MainActivity类

package com.androidbook.stockquoteclient;

import com.androidbook.stockquoteservice.IStockQuoteService;

import android.app.Activity;

import android.content.ComponentName;

import android.content.Context;

import android.content.Intent;

import android.content.ServiceConnection;

import android.os.Bundle;

import android.os.IBinder;

import android.os.RemoteException;

import android.util.Log;

import android.view.View;

import android.view.View.OnClickListener;

import android.widget.Button;

import android.widget.Toast;

public class MainActivity extends Activity {

protected static final String TAG = “StockQuoteClient”;

private IStockQuoteService stockService = null;

private Button bindBtn;

private Button callBtn;

private Button unbindBtn;

@Override

public void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.main);

bindBtn = (Button) findViewById(R.id.bindBtn);

bindBtn.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

bindService(new Intent(IStockQuoteService.class.getName()),

serConn, Context.BIND_AUTO_CREATE);

bindBtn.setEnabled(false);

callBtn.setEnabled(true);

unbindBtn.setEnabled(true);

}

});

callBtn = (Button) findViewById(R.id.callBtn);

callBtn.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

callService();

}

});

callBtn.setEnabled(false);

unbindBtn = (Button) findViewById(R.id.unbindBtn);

unbindBtn.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

unbindService(serConn);

bindBtn.setEnabled(true);

callBtn.setEnabled(false);

unbindBtn.setEnabled(false);

}

});

}

private void callService() {

try {

double val = stockService.getQuote(“SYH”);

Toast.makeText(this, "Value from service is " + val,

Toast.LENGTH_LONG).show();

} catch (RemoteException e) {

Log.e(“MainActivity”, e.getMessage(), e);

}

}

private ServiceConnection serConn = new ServiceConnection() {

// 此方法在系统建立服务连接时调用

@Override

public void onServiceConnected(ComponentName name, IBinder service) {

Log.v(TAG, “onServiceConnected() called”);

stockService = IStockQuoteService.Stub.asInterface(service);

callService();

}

// 此方法在销毁服务连接时调用

@Override

public void onServiceDisconnected(ComponentName name) {

Log.v(TAG, “onServiceDisconnected()”);

stockService = null;

}

};

}

MainActivity里边我们定义了三个按钮 分别是 Bind、Call Again 和 UnBind。当用户单击 Bind 按钮时,活动调用bindService()方法。类似地,当用户点击 UnBind 时,活动调用unbindService()方法。请注意传递给 bindService()方法 的3个参数:AIDL 服务的名称、ServiceConnection 实例和自动创建服务的标志。

我们在程序了打印了日志,我们可以看到当我们点击 Bind按钮的时候 服务端 方法的一个执行顺序。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

程序的运行效果如下

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

对于AIDL 服务,需要提供ServiceConnection接口的实现。此接口定义了两个方法:一个供系统建立服务连接时调用,另一个在销毁服务连接时调用。

最后我们在总结一下android中 本地服务和 AIDL服务的区别吧。

本地服务不支持onBind(),它从onBind()返回null,这种类型的服务只能由承载服务的应用程序组件访问。可以调用 startService()来调用本地服务。AIDL服务可以同时供 同一进程内的组件和其他应用程序的组件使用。这种类型的服务在AIDL 文件中为自身与其客户端定义一个契约。服务实现 AIDL契约,而客户端绑定到 AIDL定义。服务通过从 onBind()方法 返回AIDL接口的实现,来实现契约。客户端通过调用 bindService()来绑定到AIDL服务,并调用 unBindService()来从服务断开。

为了避免大家把aidl文件放错位置,来2张我们 程序结构的截图吧,一个是 客户端的,一个是服务端得。
客户端。。。。
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
服务端。。。
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

结语

由于篇幅限制,文档的详解资料太全面,细节内容太多,所以只把部分知识点截图出来粗略的介绍,每个小节点里面都有更细化的内容!以下是目录截图:

由于整个文档比较全面,内容比较多,篇幅不允许,下面以截图方式展示 。

再附一部分Android架构面试视频讲解:

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

结语

由于篇幅限制,文档的详解资料太全面,细节内容太多,所以只把部分知识点截图出来粗略的介绍,每个小节点里面都有更细化的内容!以下是目录截图:

[外链图片转存中…(img-HenRzPJ5-1712334303049)]

由于整个文档比较全面,内容比较多,篇幅不允许,下面以截图方式展示 。

再附一部分Android架构面试视频讲解:

[外链图片转存中…(img-4XIfapRP-1712334303049)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 21
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要为 Android AIDL 服务添加回调,您可以执行以下步骤: 1.在 AIDL 接口中定义回调方法。 2.创建一个接受回调的接口。 3.在服务中实现回调接口,并在需要时调用该接口的方法。 4.在客户端中实现回调接口,并将其传递给服务。 以下是一个简单的示例: 服务端代码: ```aidl interface IMyServiceCallback { void onUpdate(int progress); } interface IMyService { void startTask(); void registerCallback(IMyServiceCallback callback); } class MyService extends Service { private IMyServiceCallback mCallback; private final IMyService.Stub mBinder = new IMyService.Stub() { @Override public void startTask() throws RemoteException { // 启动耗时任务 for (int i = 0; i <= 100; i++) { if (mCallback != null) { mCallback.onUpdate(i); } try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } @Override public void registerCallback(IMyServiceCallback callback) throws RemoteException { mCallback = callback; } }; @Nullable @Override public IBinder onBind(Intent intent) { return mBinder; } } ``` 客户端代码: ```aidl class MyServiceConnection implements ServiceConnection { private IMyService mService; private final IMyServiceCallback.Stub mCallback = new IMyServiceCallback.Stub() { @Override public void onUpdate(int progress) throws RemoteException { // 更新进度条 } }; @Override public void onServiceConnected(ComponentName componentName, IBinder iBinder) { mService = IMyService.Stub.asInterface(iBinder); try { // 注册回调 mService.registerCallback(mCallback); // 启动任务 mService.startTask(); } catch (RemoteException e) { e.printStackTrace(); } } @Override public void onServiceDisconnected(ComponentName componentName) { mService = null; } } ``` 在这个示例中,服务端定义了一个回调接口 `IMyServiceCallback`,并在 AIDL 接口 `IMyService` 中添加了 `registerCallback()` 方法,用于向服务注册回调。服务端实现了 `registerCallback()` 方法,并在任务执行过程中调用回调接口的 `onUpdate()` 方法通知客户端。客户端实现了回调接口 `IMyServiceCallback`,并将其传递给服务端,在任务执行期间,服务端会调用客户端的 `onUpdate()` 方法通知客户端任务进度的更新。 希望这可以帮助您为 Android AIDL 服务添加回调。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值