玩转Andorid---组件篇---Service(服务)之RPC远程进程调用
在Andorid平台中,各个组件运行在自己的进程中,他们之间是不能相互访问的,但是在程序之间是不可避免的要传递一些对象,在进程之间相互通信。为了实现进程之间的相互通信,Andorid采用了一种轻量级的实现方式RPC(Remote Procedure Call 远程进程调用)来完成进程之间的通信,并且Android通过接口定义语言(Andorid Interface Definition Language ,AIDL)来生成两个进程之间相互访问的代码,例如,你在Activity里的代码需要访问Service中的一个方法,那么就可以通过这种方式来实现了。
AIDL RPC机制是通过接口来实现的,类似Windows中的COM或者Corba,但他是轻量级的,客户端和被调用实现之间是通过代理模式实现的,代理类和被代理类实现同一个接口Ibinder接口。
实例练习:
<!--[if !supportLists]-->1、 <!--[endif]-->创建一个项目,在包中创建一个AIDL文件,定义一个AIDL文件的语法和定义一个JAVA接口的语法类似,只不过文件的扩展名是“.aidl”。在AIDL文件中可以声明任一多个方法,方法可以带参数,也可以有返回值,参数和返回值是任意类型。需要注意的是,你必须导入除了内建类型(例如:int,boolean等)外的任何其他类型,即使他们在同一个包中,具体规则如下:
<!--[if !supportLists]-->① <!--[endif]-->JAVA原始类型不需要导入
<!--[if !supportLists]-->② <!--[endif]-->String,List,Map和CharSequence不需要导入
定义好的 AIDL文件可以使用ADT插件自动生成java代码
现在开始写代码:
创建一个名字为IPerson.aidl的文件
package org.hualang.rpc;
//IPerson接口
interface IPerson
{
//设置年龄
void setAge(int age);
//设置姓名方法
void setName(String name);
//显示信息方法
String display();
}
2、当创建好这个文件后,刷新工程,就会在gen目录下看到这个java接口
当你分析了这个代码后会发现,它是使用的代理模式来实现的,我们一般定义该接口的静态内部类Stub的asInterface()方法,返回我们的接口
3、实现AIDL文件生成的JAVA接口
AIDL会生成一个和.aidl文件同名的JAVA接口文件,该接口中有一个静态抽象内部类Stub,该类中声明了AIDL文件中定义的所有方法,其中有一个重要的方法是asInterface(),该方法通过代理模式返回JAVA接口的实现
我们可以定义一个实现类,PerionImpl,该类继承Stub类,实现我们定义的3个方法
/**
* @author hualang
* IPerson接口实现类
*/
package org.hualang.rpc;
import android.os.RemoteException;
public class IPersonImpl extends IPerson.Stub {
//声明两个变量
private int age;
private String name;
//显示name和age
public String display() throws RemoteException
{
return "name="+name+";age="+age;
}
@Override
public void setAge(int age) throws RemoteException {
// TODO Auto-generated method stub
this.age = age;
}
@Override
public void setName(String name) throws RemoteException {
// TODO Auto-generated method stub
this.name = name;
}
}
4、将你的接口暴露给客户端
现在我们已经实现了IPerson接口,接下俩我们要看看如何将该接口暴露给客户端调用。一般我们通过定义一个Service来实现,在Service的onBind()方法中返回该接口,当我们板顶该接口时调用该方法。
/**
* @author hualang
* 使用Service将接口暴露给客户端
*/
package org.hualang.rpc;
import org.hualang.rpc.IPerson.Stub;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
public class RemoteService extends Service {
//声明IPerson接口
private Stub iperson = new IPersonImpl();
public IBinder onBind(Intent intent)
{
return iperson;
}
}
5、客户端调用
接下来定义一个Activity来绑定远程Service,获得IPerson接口,通过RPC机制调用接口中的方法。
/**
* @author hualang
* IPerson接口实现类
*/
package org.hualang.rpc;
import android.app.Activity;
import android.app.Service;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.Toast;
public class RPCTest extends Activity {
/** Called when the activity is first created. */
private IPerson iPerson;
private Button btn;
private ServiceConnection conn = new ServiceConnection()
{
synchronized public void onServiceConnected(ComponentName name, IBinder service) {
// 获得IPerson接口
iPerson = IPerson.Stub.asInterface(service);
if (iPerson != null)
try {
// RPC 方法调用
iPerson.setName("花郎");
iPerson.setAge(22);
String msg = iPerson.display();
// 显示方法调用返回值
Toast.makeText(RPCTest.this, msg, Toast.LENGTH_LONG)
.show();
} catch (RemoteException e) {
e.printStackTrace();
}
}
@Override
public void onServiceDisconnected(ComponentName name) {
// TODO Auto-generated method stub
}
};
//声明IPerson接口
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
btn = (Button)findViewById(R.id.Button01);
btn.setOnClickListener(new OnClickListener()
{
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
Intent intent = new Intent();
intent.setAction("org.hualang.rpc.RPC_ACTION");
bindService(intent, conn, Service.BIND_AUTO_CREATE);
}
}
);
}
}
6、注意要在AndroidManifest.xml文件中注册Service
<service
android:name=".RemoteService">
<intent-filter>
<action android:name="org.hualang.rpc.RPC_ACTION"/>
</intent-filter>
</service>
运行结果如下:当点击按钮后,就会弹出远程调用后的Toast