Android中 AIDL如何使用,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。
一般创建的服务并不能被其他的应用程序访问。为了使其他的应用程序也可以访问本应用程序提供的服务,Android系统采用了远程过程调用(Remote Procedure Call,RPC)方式来实现。与很多其他的基于RPC的解决方案一样,Android使用一种接口定义语言(Interface Definition Language,IDL)来公开服务的接口。因此,可以将这种可以跨进程访问的服务称为AIDL(Android Interface Definition Language)服务。
AIDL使用简单的语法来声明接口,描述其方法以及方法的参数和返回值。这些参数和返回值可以是任何类型,甚至是其他AIDL生成的接口。
其中对于Java编程语言的基本数据类型 (int, long, char, boolean,String,CharSequence)集合接口类型List和Map,不需要import 语句。
而如果需要在AIDL中使用其他AIDL接口类型,需要import,即使是在相同包结构下。AIDL允许传递实现Parcelable接口的类,需要import
需要特别注意的是,对于非基本数据类型,也不是String和CharSequence类型的,需要有方向指示,包括in、out和inout,in表示由客户端设置,out表示由服务端设置,inout是两者均可设置。
AIDL只支持接口方法,不能公开static变量。
文件内容:
package du.pack;
interface IMyService{
//只有一个接口
String getValue();
}
public class MyService extends Service {
@Override
public IBinder onBind(Intent arg0) {
// 把内部类的对象返回给客户端使用
return new MyServiceImpl();
}
// 创建一个继承自IMyService.Stub的内部类
public class MyServiceImpl extends IMyService.Stub {
// 必须实现AIDL文件中的接口
public String getValue() throws RemoteException {
return null;
}
}
}
<service android:name=".MyService" >
<intent-filter>
<action android:name="du.pack.IMyService" />
</intent-filter>
</service>
上面的"du.pack.IMyService"是客户端用于访问AIDL服务的ID。
public class AidlClientTestActivity extends Activity {
// 远程服务端的对象
IMyService mIMyService;
private ServiceConnection mConnection = new ServiceConnection() {
public void onServiceConnected(ComponentName name, IBinder service) {
// 绑定成功,得到远程服务端的对象,目标完成!!!
mIMyService = IMyService.Stub.asInterface(service);
}
public void onServiceDisconnected(ComponentName name) {
// 解除绑定
mIMyService = null;
}
};
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
// 绑定远程服务端服务
Intent serviceIntent = new Intent("du.pack.IMyService");
bindService(serviceIntent, mConnection, Context.BIND_AUTO_CREATE);
}
}
四、Framework中使用AIDL
Framework中使用AIDL我们通过ITelephonyRegistry这个SystemService进行分析。该服务的主要作用就是对通话相关的事件进行监听,我们重心放在AIDL的实现结构上,不去关注ITelephonyRegistry具体的实现。
先来看一下这个服务的AIDL文件:
@ITelephonyRegistry.aidl
interface ITelephonyRegistry {
void listen(String pkg, IPhoneStateListener callback, int events, boolean notifyNow);
void notifyCallState(int state, String incomingNumber);
void notifyServiceState(in ServiceState state);
void notifySignalStrength(in SignalStrength signalStrength);
void notifyMessageWaitingChanged(boolean mwi);
}
再来看这个服务的真正实现:
@TelephonyRegistry.java
class TelephonyRegistry extends ITelephonyRegistry.Stub {
TelephonyRegistry(Context context) {
......
}
void listen(String pkg, IPhoneStateListener callback, int events, boolean notifyNow){
......
}
void notifyCallState(int state, String incomingNumber){
......
}
void notifyServiceState(in ServiceState state){
......
}
void notifySignalStrength(in SignalStrength signalStrength){
......
}
void notifyMessageWaitingChanged(boolean mwi){
......
}
}
上面的两个文件是这个服务的核心部分,aidl文件规定了这个服务的功能,而java文件是对功能的具体实现。但是,此时的TelephonyRegistry并没有继承Service的类,也就是说,当前他并不具备作为一个Service的资格。那么他是如何变成一个服务的呢?
在SystemService中可以找到答案。
@SystemServer.java
class ServerThread extends Thread {
@Override
public void run() {
try {
telephonyRegistry = new TelephonyRegistry(context);
ServiceManager.addService("telephony.registry", telephonyRegistry);
}
}
}
我们看到,在这一步中,把telephonyRegistry对象(也就是ITelephonyRegistry.Stub的子类对象)作为一个Service注册给了ServiceManager。并且注册的名字是“telephony.registry”
有了这一步,TelephonyRegistry就可以作为服务提供者向客户端开放了。也就是说,有了这一步,TelephonyRegistry才算上是一个真正的Service,可以接受客户端的连接申请。
那么接下来,我们怎么得到这个Service呢?
既然通过ServiceManager注册了服务,我们就需要再次通过ServiceManager得到它的服务对象。
private ITelephonyRegistry sRegistry;
sRegistry = ITelephonyRegistry.Stub.asInterface(ServiceManager.getService("telephony.registry"));
通过这样的方法,我们就得到了ITelephonyRegistry.aidl的对象sRegistry。
关于Android中 AIDL如何使用问题的解答就分享到这里了,希望以上内容可以对大家有一定的帮助,如果你还有很多疑惑没有解开,可以关注亿速云行业资讯频道了解更多相关知识。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。