Android AIDL Callback的使用(配源码)

零、示例说明

本示例,完成的功能是:客户端向服务端注册一个回调,服务端是一个商店shop,当商店里的产品 Product 有变化时,调用回调向通知客户端,什么商品更新了。

一、完整源代码

完整源码链接: https://github.com/jx0260/TestGradle

二、实现步骤

2.1 定义实体

定义 Product.aidl

// Product.aidl
package com.example.testgradle;

// Declare any non-default types here with import statements

parcelable Product;

对应的实体类:

package com.example.testgradle;

import android.os.Parcel;
import android.os.Parcelable;

public class Product implements Parcelable {

    private String no;
    private String name;

    public String getNo() {
        return no;
    }

    public void setNo(String no) {
        this.no = no;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(this.no);
        dest.writeString(this.name);
    }

    public void readFromParcel(Parcel source) {
        this.no = source.readString();
        this.name = source.readString();
    }

    public Product() {
    }

    protected Product(Parcel in) {
        this.no = in.readString();
        this.name = in.readString();
    }

    public static final Creator<Product> CREATOR = new Creator<Product>() {
        @Override
        public Product createFromParcel(Parcel source) {
            return new Product(source);
        }

        @Override
        public Product[] newArray(int size) {
            return new Product[size];
        }
    };
}

定义Callback AIDL,此 IShopProductCallback 将会注册到服务端去。

// IShopProductCallback.aidl
package com.example.testgradle;
import com.example.testgradle.Product;

// Declare any non-default types here with import statements

interface IShopProductCallback {
    oneway void onProductChanged(in Product product);
}

定义服务端 IShopAidlInterface.aidl,里面有一个registerCallback的方法

// IShopAidlInterface.aidl
package com.example.testgradle;
import com.example.testgradle.IShopProductCallback;

// Declare any non-default types here with import statements

interface IShopAidlInterface {

    String getProductInfo(int productNo);

    void buyProduct1(int productNo, String address);

    oneway void registerCallback(in IShopProductCallback callback);

}

2.2 Make project 生成相应的 AIDL实现类

客户端:
在这里插入图片描述
服务端:
在这里插入图片描述

2.2 编写客户端

Intent serviceIntent = new Intent("com.example.testgradle.ShopService");
serviceIntent.setPackage("com.example.testgradle");
bindService(serviceIntent, new ServiceConnection() {
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
        Log.i(TAG, "onServiceConnected() service=" + service);
        iShopAidlInterface = IShopAidlInterface.Stub.asInterface(service);
        try {
            Log.i(TAG, "Client: register Stub to Server!");
            iShopAidlInterface.registerCallback(new IShopProductCallback.Stub() {
                @Override
                public void onProductChanged(Product product) throws RemoteException {
                    Log.i(TAG, "Client: received product changed -- " + product.getNo() + " name:" + product.getName());
                }
            });
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }

    @Override
    public void onServiceDisconnected(ComponentName name) {

    }
}, BIND_AUTO_CREATE);

bindService() com.example.testgradle.ShopService ,在 onServiceConnected 回调中,使用拿到的 iShopAidlInterface 对象,调用 registerCallback,注意,此处传入的 callback 是 IShopProductCallback.Stub

public interface IShopProductCallback extends android.os.IInterface
{
  /** Local-side IPC implementation stub class. */
  public static abstract class Stub extends android.os.Binder

IShopProductCallback.Stub 继承了 Binder,实际上此处将 IShopProductCallback 的实现传进了参数,到达服务端。服务端拿到的是 IShopProductCallback.Stub.Proxy,这里就会有一种服务端变成客户端的感觉。实际上确实如此,在使用Callback的时候 服务端对于Callback来说 确实就变成了客户端,将变化的product 提交到服务端(也就是业务上的客户端)
此处,引用的“客户端”,“服务端”是有两套意思,一套是指我们总体业务的 客户端、服务端,另一套是指对于callback来说,程序实现的 客户端、服务端。

2.3 编写服务端

package com.example.testgradle;

import android.app.Service;
import android.content.Intent;
import android.os.Handler;
import android.os.IBinder;
import android.os.RemoteException;
import android.util.Log;

import javax.security.auth.login.LoginException;

public class ShopService extends Service {
    private String TAG = "ShopService";
    private IShopProductCallback mCallback;

    public ShopService() {
    }

    private IShopAidlInterface.Stub mBinder = new IShopAidlInterface.Stub() {
        @Override
        public void registerCallback(IShopProductCallback callback) throws RemoteException {
            Log.i(TAG, "Server: registerCallback() this callback obj is a proxy!");
            mCallback = callback;

            mHandler.postDelayed(new Runnable() {
                @Override
                public void run() {
                    Product product = new Product();
                    product.setNo("No.0001");
                    product.setName("banana");
                    try {
                        Log.i(TAG, "Server: the proxy obj invoke onProductChanged, pass the product to the client!");
                        mCallback.onProductChanged(product);
                    } catch (RemoteException e) {
                        e.printStackTrace();
                    }
                }
            }, 20000);
        }
    };

    @Override
    public IBinder onBind(Intent intent) {
        return mBinder;
    }
}

服务端 public void registerCallback(IShopProductCallback callback) 拿到客户端传来的 callback其实是一个proxy 存到mCallback 本地变量中,20秒后,创建一个变化的Product 调用 mCallback.onProductChanged(product); 将商品信息回调给客户端。

2.4 运行

日志打印:

com.example.testgradle I/ShopService: Server: registerCallback() this callback obj is a proxy!

com.example.testgradle I/ShopService: Server: the proxy obj invoke onProductChanged, pass the product to the client!
com.example.clientdemo I/ClientActivity: Client: received product changed -- No.0001 name:banana
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 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
发出的红包

打赏作者

洛克Lee

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值