編輯:關於Android編程
package com.example.advanceandroid.aidl;
interface ILogin {
String login();
}
注意看,接口和方法聲明都不用public,方法加入public會提示錯誤。編寫完後如果eclipse開啟了自動編譯則會在gen/com.example.advanceandroid.aidl下生成一個ILogin.java類,內容大致如下:
package com.example.advanceandroid.aidl;
public interface ILogin extends android.os.IInterface
{
/** Local-side IPC implementation stub class. */
public static abstract class Stub extends android.os.Binder implements
com.example.advanceandroid.aidl.ILogin
{
private static final java.lang.String DESCRIPTOR = "com.example.advanceandroid.aidl.ILogin";
/** Construct the stub at attach it to the interface. */
public Stub()
{
this.attachInterface(this, DESCRIPTOR);
}
/**
* Cast an IBinder object into an com.example.advanceandroid.aidl.ILogin
* interface, generating a proxy if needed.
*/
public static com.example.advanceandroid.aidl.ILogin asInterface(android.os.IBinder obj)
{
if ((obj == null)) {
return null;
}
android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
if (((iin != null) && (iin instanceof com.example.advanceandroid.aidl.ILogin))) {
return ((com.example.advanceandroid.aidl.ILogin) iin);
}
return new com.example.advanceandroid.aidl.ILogin.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
{
switch (code)
{
case INTERFACE_TRANSACTION: {
reply.writeString(DESCRIPTOR);
return true;
}
case TRANSACTION_login: { // 1、登錄請求,執行的是this.login();
data.enforceInterface(DESCRIPTOR);
java.lang.String _result = this.login();
reply.writeNoException();
reply.writeString(_result);
return true;
}
}
return super.onTransact(code, data, reply, flags);
}
private static class Proxy implements com.example.advanceandroid.aidl.ILogin
{
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.lang.String login() throws android.os.RemoteException // 2、Proxy中的login,通過Binder機制實現IPC
{
android.os.Parcel _data = android.os.Parcel.obtain();
android.os.Parcel _reply = android.os.Parcel.obtain();
java.lang.String _result;
try {
_data.writeInterfaceToken(DESCRIPTOR);
mRemote.transact(Stub.TRANSACTION_login, _data, _reply, 0);
_reply.readException();
_result = _reply.readString();
} finally {
_reply.recycle();
_data.recycle();
}
return _result;
}
}
static final int TRANSACTION_login = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
}
public java.lang.String login() throws android.os.RemoteException;
} 可以看到,該類中自動生成了ILogin接口,該接口中又一個login()函數。但最重要的是裡面生成了一個Stub類,該類集成子Binder類,並且實現了ILogin接口。Stub裡面最重要的就是asInterface()這個函數,在這個函數中會判斷obj參數的類型,如果是該obj是本地的接口類似,則認為不是IPC,會將該obj轉換成ILogin類型;否則會通過自動生成的另一個內部類Proxy來包裝obj,將其賦值給Proxy中的mRemote屬性。Proxy類也實現了ILogin接口,在login()函數中,Proxy將通過Binder機制向服務端傳遞請求和數據,如上面代碼中的注釋2。這是客戶端的工作算是完成了。服務端還沒有完,最重要的一步時建立一個Service,內容大致如下 :
/**
* AIDL服務端接口,LoginStubImpl實現了ILogin接口.
*
* @author mrsimple
*/
public class LoginService extends Service {
/**
*
*/
IBinder mBinder = new LoginStubImpl();
/**
* @author mrsimple
*/
class LoginStubImpl extends Stub {
@Override
public String login() throws RemoteException {
return "這是從 " + this.getClass().getName() + " 返回的字符串";
}
}
/*
* 返回Binder實例,即實現了ILogin接口的Stub的子類,這裡為LoginStubImpl
* [url=home.php?mod=space&uid=133757]@see[/url] android.app.Service#onBind(android.content.Intent)
*/
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
} 該Service我們這裡命名為LoginService,繼承自Service,然後建一個名為LoginServiceImpl的內部類,該類繼承自自動生成的Stub,然後實現login()方法。在LoginService中聲明一個IBinder字段mBinder : IBinder mBinder = new LoginStubImpl();並且在LoginService的onBind函數中將mBinder對象返回。即在客戶端建立與服務端的連接時,會調用onBind方法將mBinder對象返回,在客戶端的ServiceConnection類的onServiceConnected函數中得到的對象IBinder就是經過BinderProxy包裝的LoginService中的mBinder對象。因此在服務端中的onTransact中調用的this.login()函數實際上就是調用的LoginStubImpl中的login()函數。
在服務端程序的AndroidManifest.xml中注冊LoginService,如下 :
在Activity中加入如下代碼 :
ServiceConnection mLoginConnection = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
Log.d("", "### aidl disconnected.");
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
Log.d("", "### aidl onServiceConnected. service : " + service.getClass().getName());
ILogin login = Stub.asInterface(service);
Log.d("", "### after asInterface : " + login.getClass().getName());
try {
Log.d("", "### login : " + login.login());
// Toast.makeText(MainActivity.this, "onServiceConnected : " +
// login.login(),
// Toast.LENGTH_SHORT).show();
} catch (RemoteException e) {
e.printStackTrace();
}
}
};
@Override
protected void onResume() {
super.onResume();
// 服務端的action
Intent aidlIntent = new Intent("com.example.advanceandroid.aidl.LoginService");
bindService(aidlIntent, mLoginConnection, Context.BIND_AUTO_CREATE);
}
@Override
protected void onStop() {
super.onStop();
// unbind
unbindService(mLoginConnection);
}運行先運行服務端程序,然後在啟動客戶端程序,可以看到客戶端輸出如下Log:
09-02 10:40:54.662: D/(9589): ### aidl onServiceConnected. service : android.os.BinderProxy 09-02 10:40:54.662: D/(9589): ### after asInterface : com.example.advanceandroid.aidl.ILogin$Stub$Proxy 09-02 10:40:54.662: D/(9589): ### login : 這是從 com.example.advanceandroid.aidl.LoginService$LoginStubImpl 返回的字符串可以看淡onServiceConnected(ComponentName name, IBinder service)中的service對象是BinderProxy類型,經過asInterface轉換後被包裝成了Proxy類型,但是調用的時候,執行的是服務端LoginStubImpl中的login()函數。因此,LoginStubImpl實例mBinder被服務端包裝成BinderProxy類型,再經過客戶端的Proxy進行包裝,通過Binder機制進行數據傳輸,實現IPC。
Android-版本控制 SVN & GIT 環境安裝
一 、 虛擬機的安裝常見的虛擬機產品有 VMware 公司的 VMware Workstation、Oracle 公司的 VirtualBox。因為 VMware 體積相
Android自定義View之三種流行進度條的寫法
概述:利用自定義View的onDraw()方法,可以繪制很多種圖形,進度框只是其中之一。Demo這是一個模擬下載的demo。自中央逐漸充滿型圓形進度框demo1publi
安卓(Android) 自定義控件(View) 新手教學
作為一只安卓自學的小白,今天第一天發表微博還是有點小激動的,好了,廢話少說下面開始我的安卓自定義控件知識總結。我的demo是一個自定義的TopBar,左邊一個Button
Android源碼解析--SwipeMenuListView仿QQ聊天左滑
緒論:好久沒寫博客了,最近比較懶,不想寫博客,但是在看書,看一些Android進階的書,這裡小編也給大家推薦幾本適合進階的書,相信會對你有所幫助的。1.《Android群