关于AIDL和Binder

本文详细介绍了Android中实现跨进程通信的AIDL(Android Interface Definition Language)和Binder机制。AIDL基于Binder,用于创建进程间通信的接口。内容包括AIDL的通信机制解析、AIDL的使用步骤及示例,以及直接继承Binder实现服务端和客户端的通信方式。


AIDL是一种跨进程通信的方式,通信基于Binder。

直接继承Binder也可以实现跨进程通信。

Binder

Binder是一个类,它实现了IBinder接口,而IBinder接口定义了与远程对象的交互协议。通常在进行跨进程通信时,不需要实现IBinder接口,直接从Binder派生即可。

除了实现IBinder接口外,Binder中还提供了两个重要的接口。

  • transact(),客户端调用,用于发送调用请求
  • onTransact(),服务端响应,用于接收调用请求

因为以上的原因,Binder成为了客户端与服务端的通信媒介,其主要用在Service组件应用中。

AIDL

机制

AIDL的通信基于Binder。

下面写一个AIDL的demo讲解他的通信机制。

新建Book.aidl文件

package com.competition.pdking.ipcdemo;

parcelable Book;

然后创建一个IBookManager.aidl文件

// IBookManager.aidl
package com.competition.pdking.ipcdemo;

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

import com.competition.pdking.ipcdemo.Book;
interface IBookManager {

    List<Book> getBookList();
    void addBook(in Book book);
}

Android会生成一个java文件

/*
 * This file is auto-generated.  DO NOT MODIFY.
 * Original file: C:\\Projects\\AndroidProjects\\AndroidDemoProject\\IPCDemo\\app\\src\\main
 * \\aidl\\com\\competition\\pdking\\ipcdemo\\IBookManager.aidl
 */
package com.competition.pdking.ipcdemo;

public interface IBookManager extends android.os.IInterface {
    /**
     * Local-side IPC implementation stub class.
     */
    public static abstract class Stub extends android.os.Binder implements com.competition.pdking.ipcdemo.IBookManager {
        private static final java.lang.String DESCRIPTOR = "com.competition.pdking.ipcdemo" +
                ".IBookManager";

        /**
         * Construct the stub at attach it to the interface.
         */
        public Stub() {
            this.attachInterface(this, DESCRIPTOR);
        }

        /**
         * Cast an IBinder object into an com.competition.pdking.ipcdemo.IBookManager interface,
         * generating a proxy if needed.
         */
        public static com.competition.pdking.ipcdemo.IBookManager asInterface(android.os.IBinder obj) {
            if ((obj == null)) {
                return null;
            }
            android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
            if (((iin != null) && (iin instanceof com.competition.pdking.ipcdemo.IBookManager))) {
                return ((com.competition.pdking.ipcdemo.IBookManager) iin);
            }
            return new com.competition.pdking.ipcdemo.IBookManager.Stub.Proxy(obj);
        }

        @Override
        public android.os.IBinder asBinder() {
            return this;
        }

        @Override
        public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply,
                                  int flags) throws android.os.RemoteException {
            java.lang.String descriptor = DESCRIPTOR;
            switch (code) {
                case INTERFACE_TRANSACTION: {
                    reply.writeString(descriptor);
                    return true;
                }
                case TRANSACTION_getBookList: {
                    data.enforceInterface(descriptor);
                    java.util.List<com.competition.pdking.ipcdemo.Book> _result =
                            this.getBookList();
                    reply.writeNoException();
                    reply.writeTypedList(_result);
                    return true;
                }
                case TRANSACTION_addBook: {
                    data.enforceInterface(descriptor);
                    com.competition.pdking.ipcdemo.Book _arg0;
                    if ((0 != data.readInt())) {
                        _arg0 = com.competition.pdking.ipcdemo.Book.CREATOR.createFromParcel(data);
                    } else {
                        _arg0 = null;
                    }
                    this.addBook(_arg0);
                    reply.writeNoException();
                    return true;
                }
                default: {
                    return super.onTransact(code, data, reply, flags);
                }
            }
        }

        private static class Proxy implements com.competition.pdking.ipcdemo.IBookManager {
            private android.os.IBinder mRemote;

            Proxy(android.os.IBinder remote) {
                mRemote = remote;
            }

            @Override
            public android.os.IBinder asBinder() {
                return mRemote;
            }

            public java.lang.String getInterfaceDescriptor() {
                return DESCRIPTOR;
            }

            @Override
            public java.util.List<com.competition.pdking.ipcdemo.Book> getBookList() throws android.os.RemoteException {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                java.util.List<com.competition.pdking.ipcdemo.Book> _result;
                try {
                    _data.writeInterfaceToken(DESCRIPTOR);
                    mRemote.transact(Stub.TRANSACTION_getBookList, _data, _reply, 0);
                    _reply.readException();
                    _result = _reply.createTypedArrayList(com.competition.pdking.ipcdemo.Book.CREATOR);
                } finally {
                    _reply.recycle();
                    _data.recycle();
                }
                return _result;
            }

            @Override
            public void addBook(com.competition.pdking.ipcdemo.Book book) throws android.os.RemoteException {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                try {
                    _data.writeInterfaceToken(DESCRIPTOR);
                    if ((book != null)) {
                        _data.writeInt(1);
                        book.writeToParcel(_data, 0);
                    } else {
                        _data.writeInt(0);
                    }
                    mRemote.transact(Stub.TRANSACTION_addBook, _data, _reply, 0);
                    _reply.readException();
                } finally {
                    _reply.recycle();
                    _data.recycle();
                }
            }
        }

        static final int TRANSACTION_getBookList = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
        static final int TRANSACTION_addBook = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1);
    }

    public java.util.List<com.competition.pdking.ipcdemo.Book> getBookList() throws android.os.RemoteException;

    public void addBook(com.competition.pdking.ipcdemo.Book book) throws android.os.RemoteException;
}

  • IBookManager.java这个类他继承IInterFace接口,同时他也是一个接口,所有可以在Binder中传输的接口都需要继承IInterface接口。

  • 首先他声明了getBookList和addBook两个方法。这显然就是我们在aidl文件中声明的方法。同时还声明了两个id用于标记这两个方法。

  • 然后,他声明了一个内部类Stub,这个类就是一个Binder类,当位于一个进程时,方法调用不会走跨进程的transact过程;当位于不同的进程时,会走transact过程,这个逻辑由Stub的内部类Proxy完成。

然后逐个解释作用:

  • DESCRIPTOR——Binder的唯一标识,用来表示当前Binder的类名表示。
  • asInterface——将服务端的Binder对象转换为客户端所需的AIDL接口类型,这个转换过程是区分进程的。如果位于同一个进程那么此方法返回的就是服务端的Stub对象本身,否则返回系统封装的Stub.Proxy对象。
  • asBinder——返回当前Binder对象。
  • onTransact——这个方法运行在服务端的Binder线程池中,当客户端发起远程请求后,会经系统底层封装后交此方法来处理。 public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) 服务端通过code可以确定客户端请求的目标方法,接着从data中取出目标方法的参数,然后执行目标方法,目标方法执行完后,就向reply中写入返回值。
  • Proxy#getBookList——这个方法运行在客户端,调用流程如下:创建Parcel对象_data和_reply,接着调用transact方法发起远程调用请求,同时线程挂起,然后服务端的onTransact方法会调用,知道调用过程返回,当前线程继续执行,并从_reply中取出返回的结果,最后返回结果。
  • Proxy#addBook——和getBookList过程一样,只不过没有返回值。

使用

服务端

/**
 * @author liupeidong
 * Created on 2019/11/7 21:40
 */
public class MyService extends Service {

    IBookManager.Stub mBinder = new IBookManager.Stub() {
        @Override
        public List<Book> getBookList() throws RemoteException {
            return null;
        }

        @Override
        public void addBook(Book book) throws RemoteException {

        }
    };

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

客户端

public class MainActivity extends AppCompatActivity {

    IBookManager aidl;

    ServiceConnection connection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            aidl = IBookManager.Stub.asInterface(service);
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {

        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Intent intent = new Intent(this, MyService.class);
        bindService(intent, connection, Context.BIND_AUTO_CREATE);

    }
}

这样就通过AIDL完成了进程间通信。

继承Binder实现继承间通信

服务端

服务端只要继承一个Binder,并完成重写onTransact方法,在onTransact方法里面对不同的方法进行判断即可。

	private MyBinder mBinder = new MyBinder();
 
	private class MyBinder extends Binder
	{
		@Override
		protected boolean onTransact(int code, Parcel data, Parcel reply,
				int flags) throws RemoteException
		{
			switch (code)
			{
			case 0x110:
			{
				data.enforceInterface(DESCRIPTOR);
				int _arg0;
				_arg0 = data.readInt();
				int _arg1;
				_arg1 = data.readInt();
				int _result = _arg0 * _arg1;
				reply.writeNoException();
				reply.writeInt(_result);
				return true;
			}
			case 0x111:
			{
				data.enforceInterface(DESCRIPTOR);
				int _arg0;
				_arg0 = data.readInt();
				int _arg1;
				_arg1 = data.readInt();
				int _result = _arg0 / _arg1;
				reply.writeNoException();
				reply.writeInt(_result);
				return true;
			}
			}
			return super.onTransact(code, data, reply, flags);
		}
 
	};
 

客户端

创建ServiceConnection 绑定IBinder

	private IBinder mPlusBinder;
	private ServiceConnection mServiceConnPlus = new ServiceConnection()
	{
		@Override
		public void onServiceDisconnected(ComponentName name)
		{
			Log.e("client", "mServiceConnPlus onServiceDisconnected");
		}
 
		@Override
		public void onServiceConnected(ComponentName name, IBinder service)
		{
 
			Log.e("client", " mServiceConnPlus onServiceConnected");
			mPlusBinder = service;
		}
	}; 

然后调用Binder的方法时需要注意,需要加上自己的方法标志或ID来区分不同的方法,然后的流程就可AIDL的差不多了,创建Parcel _data 和_reply,接着调用transact方法进行远程请求,最后从_reply读取结果即可。

	public void mulInvoked(View view)
	{
 
		if (mPlusBinder == null)
		{
			Toast.makeText(this, "未连接服务端或服务端被异常杀死", Toast.LENGTH_SHORT).show();
		} else
		{
			android.os.Parcel _data = android.os.Parcel.obtain();
			android.os.Parcel _reply = android.os.Parcel.obtain();
			int _result;
			try
			{
				_data.writeInterfaceToken("CalcPlusService");
				_data.writeInt(50);
				_data.writeInt(12);
				mPlusBinder.transact(0x110, _data, _reply, 0);
				_reply.readException();
				_result = _reply.readInt();
				Toast.makeText(this, _result + "", Toast.LENGTH_SHORT).show();
 
			} catch (RemoteException e)
			{
				e.printStackTrace();
			} finally
			{
				_reply.recycle();
				_data.recycle();
			}
		}
 
	}
	
	public void divInvoked(View view)
	{
 
		if (mPlusBinder == null)
		{
			Toast.makeText(this, "未连接服务端或服务端被异常杀死", Toast.LENGTH_SHORT).show();
		} else
		{
			android.os.Parcel _data = android.os.Parcel.obtain();
			android.os.Parcel _reply = android.os.Parcel.obtain();
			int _result;
			try
			{
				_data.writeInterfaceToken("CalcPlusService");
				_data.writeInt(36);
				_data.writeInt(12);
				mPlusBinder.transact(0x111, _data, _reply, 0);
				_reply.readException();
				_result = _reply.readInt();
				Toast.makeText(this, _result + "", Toast.LENGTH_SHORT).show();
 
			} catch (RemoteException e)
			{
				e.printStackTrace();
			} finally
			{
				_reply.recycle();
				_data.recycle();
			}
		}
 
	}

<think> 我们正在讨论AIDLBinder的区别。根据引用内容,我们可以总结如下: 1. AIDL(Android Interface Definition Language)是一种接口定义语言,用于定义跨进程通信(IPC)的接口。它允许你定义客户端服务端达成一致的编程接口,以便进行进程间通信。 2. Binder是Android中实现跨进程通信的底层机制。它是一个驱动,也是一个类(实现了IBinder接口)。Binder框架负责在进程间传递消息,包括数据的序列化、传输反序列化。 区别: - AIDL是一种语言工具,用于定义接口;而Binder是实际进行通信的框架驱动。 - 使用AIDL时,系统会自动生成基于Binder的代码。因此,AIDL可以看作是Binder的一个封装,简化了Binder的使用。 - Binder是更底层的实现,开发者可以直接使用Binder进行IPC,但这样需要处理更多细节。而AIDL通过自动生成代码,让开发者专注于接口定义。 因此,简单说:AIDL是接口定义语言,Binder是通信机制。AIDL依赖于Binder来实现其功能。 参考文献: [^1]: Android中的binderaidl机制 [^2]: Binder AIDL 理解 §§相关问题§§ 1. AIDL支持哪些数据类型? 2. Binder在Android中除了IPC还有哪些作用? 3. 如何不使用AIDL而直接使用Binder进行跨进程通信? 4. 使用AIDL时,如何确保数据的安全性? </think>### AIDLBinder的区别分析 AIDL(Android Interface Definition Language)Binder是Android跨进程通信(IPC)的核心组件,二者关系紧密但存在本质差异: | **特性** | **AIDL** | **Binder** | |------------------|---------------------------------------|-----------------------------------------| | **本质** | 接口定义语言 | 底层IPC通信框架(继承`IBinder`的对象) | | **作用** | 声明跨进程通信的接口规范 | 实现跨进程方法调用数据传输 | | **抽象层级** | 高层抽象(面向开发者) | 底层实现(面向系统) | | **使用方式** | 通过`.aidl`文件定义接口 | 通过`Binder`对象实现RPC协议 | | **代码生成** | 自动生成`Stub``Proxy`类 | 需手动实现`onTransact()`等核心方法 | | **开发复杂度** | 低(自动处理序列化/反序列化) | 高(需手动处理数据编解码) | #### 具体区别说明: 1. **角色定位不同** - AIDL是**接口描述语言**,用于标准化跨进程接口(如方法签名、参数类型)。开发者只需定义接口,编译器会自动生成跨进程通信代码[^1]。 - Binder是**通信执行者**,负责实际的数据传输。它通过内核驱动完成进程间内存共享线程调度[^2]。 2. **实现原理差异** - AIDL生成的`Stub`类继承`Binder`并重写`onTransact()`,自动将接口调用转化为Binder操作: ```java // 自动生成的Stub类(简化) public static abstract class Stub extends Binder implements IMyInterface { @Override protected boolean onTransact(int code, Parcel data, Parcel reply, int flags) { switch (code) { case TRANSACTION_add: // 自动处理add()方法调用 int arg0 = data.readInt(); int arg1 = data.readInt(); int result = this.add(arg0, arg1); reply.writeInt(result); return true; } return super.onTransact(code, data, reply, flags); } } ``` - 直接使用Binder需手动实现上述过程,包括参数序列化、事务码分配等。 3. **适用场景** - **AIDL**:适合常规IPC需求(如服务调用),开发效率高。 - **Binder**:需自定义RPC协议或优化性能时使用(如系统服务)。 > 总结:AIDL是**声明层工具**,Binder是**传输层基础**。AIDL通过自动生成Binder代码简化开发,而直接操作Binder则更灵活但更复杂[^1][^2]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值