大概实现类似TelephoneManager这样的远程服务,但是由于TelephoneManager是已经封装过的代理类,所以我们不需要去获取binder,但是对于调用另一个应用的service,需要通过aidl来通信。请注意:这两种实现方式会有区别,后面会附上代码。

这里主要讲前者,对于aidl实现跨进程调用大概介绍一下。

1.添加service到SystemService


IInnerToolService  定义接口,InnerToolService实现该接口
IInnerToolCB.aidl  定义回调  不熟悉aidl的话可以先不加



1).添加需要的Service的java文件。

注意:包名也要一样。package com.android.server;然后找到SystemServer,修改如下在ServerThread的run方法里添加:

//add innertool
try {              
    ServiceManager.addService("innertool", new InnerToolService(context));            
} catch (Throwable e) {     
}
//End

注意"innertool"相当于service的标记,在使用getSystemService来获取时需要提供该标记。

2).frameworks\base\core\java\android\app\ContextImpl.java


这里主要是响应getSystemService,如果service未启动,则启动service。

首先添加import语句
			//inner tool
			import com.android.innertool.IInnerToolService;
			//End



然后在内部类ServiceFetcher的static语句块里添加以下代码:
			
			//add innertool 
			registerService("innertool", new ServiceFetcher() {
                public Object getService(ContextImpl ctx) {
			IBinder b = ServiceManager.getService("innertool");
			IInnerToolService service = IInnerToolService.Stub.asInterface(b);
			Log.e("tool", "fetch innertool service = " + service);
			return  service;
                }});
			//End
3).aidl相关文件的编译
framework/base/Android.mk文件,在LOCAL_SRC_FILES下添加相关的aidl文件
		
		LOCAL_SRC_FILES += \
				innertool/java/com/android/innertool/IInnerToolService.aidl \
				innertool/java/com/android/innertool/IInnerToolCB.aidl

到这里基本就添加完成了,可能部分不同的平台或者项目会存在差异,大家需要细心。然后是对于远程service的调用

假设aidl定义的接口如下

interface IInnerToolService{
	void exec(String arg0,String arg1,IInnerToolCB arg2);
}



2.通过aidl调用远程service

上面定义了aidl文件,使用的包名是com.android.innertool,那我们本地应用也需要有这个包名下的aidl文件

即测试应用需要创建com.android.innertool这个包,并将IInnerToolService.aidl和IInnerToolCB.aidl文件放进去,这样会在gen/生成对应的java文件。

然后就可以使用该SystemService的接口了。

private IInnerToolService toolService2;
toolService2 = (IInnerToolService) getSystemService("innertool");
try {
    toolService2.exec("aaa", "bbb", new InnerToolCB());
} catch (RemoteException e1) {
    // TODO Auto-generated catch block
    e1.printStackTrace();
}


这样基本就走通了简单的调用流程。

附上相应的代码

package com.android.server;

import android.content.Context;
import android.os.RemoteException;
import android.util.Log;

import com.android.innertool.IInnerToolCB;
import com.android.innertool.IInnerToolService;

public class InnerToolService extends IInnerToolService.Stub{

	private Context mContext;

	public InnerToolService(Context context) {	
		mContext = context; 
	}

	@Override
	public void exec(String arg0, String arg1, IInnerToolCB arg2)
			throws RemoteException {
		// TODO Auto-generated method stub
		Log.v("InnerToolService", "exec,"+arg0+","+arg1+","+arg2);
	}
}




3.下面简单的介绍调用其他应用的service

假设仍然以上面的接口作为测试,这样被调用的应用需要有一个Serivce来实现IInnerToolService.Stub并返回该binder

代码如下:

package com.android.test;

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

public class MyService extends Service{

	private IInnerToolService.Stub binder = new IInnerToolService.Stub(){

		@Override
		public void exec(String arg0, String arg1, IInnerToolCB arg2)
				throws RemoteException {
			// TODO Auto-generated method stub
			Log.v("InnerToolService", "exec,"+arg0+","+arg1+","+arg2);
		}
	};

	@Override
	public IBinder onBind(Intent intent) {
		// TODO Auto-generated method stub
		Log.v("InnerToolService", "onBind");
		return binder;
	}
}



这样调用的地方就可以通过bindService来获取该binder从而能跨进程通信

调用代码

private IInnerToolService toolService; 
private ServiceConnection mSerConn;
mSerConn = new ServiceConnection() {


@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// TODO Auto-generated method stub

toolService = IInnerToolService.Stub.asInterface(service);
}


@Override
public void onServiceDisconnected(ComponentName name) {
// TODO Auto-generated method stub

}

};
try {
toolService.exec("aaa", "bbbb", new InnerToolCB());
} catch (RemoteException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}



通过两者对比,可以看到都是通过binder来通信的,不过由于获取binder的方式不同,所以代码有些区别。



4.aidl编译相关


如果是简单的接口文件,在mk文件里附上路径就可以,如果是Parcel类型,则需要实现java文件。因为aidl文件里只有一句声明


package com.android.innertool;

parcelable Entity;

对于这个例子,则需要在com.android.innertool下创建Entity.java,并implements Parcelable,然后按需求设计相应的数据结构等。


另外注意,必须实现以下方法


public void writeToParcel(Parcel dest, int arg1) 
public void readFromParcel(Parcel _reply)  // 这是系统解析aidl并生成java文件时在接口里要调用的

对于第二个方法,大家也许会发现,平时使用intent传递的类型不需要重写啊,只需要实现Parcelable.Creator<AppAction> CREATOR


但是这里必须要实现,因为在onTransact方法里会通过这个方法来获取参数


_arg0 = com.android.innertool.Entity.CREATOR.createFromParcel(data);



然后在接口里使用这个类时,需要添加限定符in,out,或者inout,否则编译不过。


例如在上面的aidl文件IInnerToolService里添加接口send(inout Entity entity),这里的in,out不明确的话,使用inout。