AndroidのAIDL使用時のクロスプロセスコール-ServerコールバックClient
まずserver側に2つのaidlファイルを作成します
ITaskCallback.aidlコールバッククライアントを格納する方法
ITaskBinder.aidl供給クライアント側呼び出しを格納する方法
次に、サービス側を確立し、コールバックを実現するには、RemoteCallbackListオブジェクト(コールバックオブジェクトを格納するリスト)を新規作成し、ブロードキャストを送信するような形式でコールバックを実現する必要がある
MyService.java
最後にクライアントを確立しaidlを用いてコールバックメソッドを実現する.
MyActivity.java
aidlの使い方をまとめてみます
AIDLの作成方法:AIDL構文は簡単で、1つ以上のメソッドを持つインタフェースを宣言したり、パラメータと戻り値を渡したりすることができます.これらのパラメータと戻り値は、リモートコールの必要性から任意のタイプではない.以下はAIDLがサポートするデータ型です:1.import宣言を必要としない簡単なJavaプログラミング言語タイプ(int,booleanなど)2.String,CharSequenceは特別な声明を必要としない.リスト、Map、Parcelablesのタイプである、これらのタイプに含まれるデータメンバーは、単純なデータ型、Stringなどの他の比でサポートされているタイプのみである.(また、Parcelablesを試していないので、Eclipse+ADTでコンパイルしていますが、後でサポートされるかもしれません).以下はAIDL構文://ファイル名:SomeClass.aidl//ファイルにはコメントがあります.javaと同じ//package以前のコメントは無視されます.//関数と変数の以前の注釈は、本番javaコードに追加されます.package com.cmcc.demo;//import導入文import com.cmcc.demo.ITaskCallback;interface ITaskBinder{//関数はjavaと同様に0から複数のパラメータがあり、1つの戻り値boolean isTaskRunning();void stopRunningTask();//パラメータは別のaidl定義インタフェースvoid registerCallback(ITaskCallback cb);void unregisterCallback(ITaskCallback cb);void unregisterCallback(ITaskCallback cb);//パラメータはString、in表入入力タイプ、outで出力タイプを表すことができる. int getCustomerList(in String branch, out String customerList); }
インタフェースを実現するにはいくつかの原則がある.投げ出された異常は呼び出し者に返さないでください.プロセス間放出異常処理は望ましくない. .IPC呼び出しは同期されます.IPCサービスが完了するのに数ミリ秒以上かかることを知っている場合は、Activityのプライマリ・スレッドで呼び出されるのを避ける必要があります.つまりIPC呼び出しはアプリケーションを保留することによってインタフェースの応答を失う.この場合、単一のスレッドで処理することを考慮すべきである. .AIDLインタフェースで静的属性を宣言することはできません.IPCの呼び出し手順:1.インタフェースタイプの変数を宣言します.aidlファイルで定義します. 2. ServiceConnectionを実装します. 3. ApplicationContextを呼び出す.bindService()は、ServiceConnection実装において伝達する. 4. サービス接続onServiceConnected()インプリメンテーションでは、IBinderインスタンス(呼び出されたService)が受信されます.YourInterfaceName.Stub.asInterface(IBinder)service)は、パラメータをYourInterfaceタイプに変換します. 5. インタフェースで定義されたメソッドを呼び出します.接続が切断されたときに投げ出されるDeadObjectExceptionの異常を常に検出します.リモートメソッドによってのみ放出されます. 6. 接続を切断し、インタフェースインスタンスのApplicationContextを呼び出す.unbindService()
ITaskCallback.aidlコールバッククライアントを格納する方法
package com.cmcc.demo.server;
interface ITaskCallback {
void actionPerformed(int actionId);
}
ITaskBinder.aidl供給クライアント側呼び出しを格納する方法
package com.cmcc.demo.server;
import com.cmcc.demo.server.ITaskCallback;
interface ITaskBinder {
boolean isTaskRunning();
void stopRunningTask();
void registerCallback(ITaskCallback cb);
void unregisterCallback(ITaskCallback cb);
}
次に、サービス側を確立し、コールバックを実現するには、RemoteCallbackListオブジェクト(コールバックオブジェクトを格納するリスト)を新規作成し、ブロードキャストを送信するような形式でコールバックを実現する必要がある
MyService.java
package com.cmcc.demo.server;
import com.cmcc.demo.server.ITaskBinder;
import com.cmcc.demo.server.ITaskCallback;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteCallbackList;
import android.os.RemoteException;
import android.util.Log;
public class MyService extends Service {
private static final String TAG = "aidltest";
@Override
public void onCreate() {
printf("service create");
}
@Override
public void onStart(Intent intent, int startId) {
printf("service start id=" + startId);
callback(startId);
}
@Override
public IBinder onBind(Intent t) {
printf("service on bind");
return mBinder;
}
@Override
public void onDestroy() {
printf("service on destroy");
super.onDestroy();
}
@Override
public boolean onUnbind(Intent intent) {
printf("service on unbind");
return super.onUnbind(intent);
}
public void onRebind(Intent intent) {
printf("service on rebind");
super.onRebind(intent);
}
private void printf(String str) {
Log.v(TAG, "###################------ " + str + "------");
}
void callback(int val) {
final int N = mCallbacks.beginBroadcast();
for (int i=0; imCallbacks = new RemoteCallbackList ();
}
最後にクライアントを確立しaidlを用いてコールバックメソッドを実現する.
MyActivity.java
package com.cmcc.demo;
import android.app.Activity;
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.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import com.cmcc.demo.server.*;
public class MyActivity extends Activity {
private static final String TAG = "aidltest";
private Button btnOk;
private Button btnCancel;
@Override
public void onCreate(Bundle icicle) {
super.onCreate(icicle);
setContentView(R.layout.test_service);
btnOk = (Button)findViewById(R.id.btn_ok);
btnCancel = (Button)findViewById(R.id.btn_cancel);
btnOk.setText("Start Service");
btnCancel.setText("Stop Service");
btnCancel.setEnabled(false);
btnOk.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
onOkClick();
}
});
btnCancel.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
onCancelClick();
}
});
}
void onOkClick() {
printf("send intent to start");
Bundle args = new Bundle();
Intent intent = new Intent("com.cmcc.demo.IMyService");
intent.putExtras(args);
startService(intent);
//bindService(intent, mConnection, Context.BIND_AUTO_CREATE);
btnCancel.setEnabled(true);
}
void onCancelClick() {
printf("send intent to stop");
//unbindService(mConnection);
Intent intent = new Intent("com.cmcc.demo.IMyService");
stopService(intent);
btnCancel.setEnabled(false);
}
private void printf(String str) {
Log.v(TAG, "###################------ " + str + "------");
}
ITaskBinder mService;
private ServiceConnection mConnection = new ServiceConnection() {
public void onServiceConnected(ComponentName className, IBinder service) {
mService = ITaskBinder.Stub.asInterface(service);
try {
mService.registerCallback(mCallback);
} catch (RemoteException e) {
}
}
public void onServiceDisconnected(ComponentName className) {
mService = null;
}
};
private ITaskCallback mCallback = new ITaskCallback.Stub() {
public void actionPerformed(int id) {
printf("callback id=" + id);
}
};
}
aidlの使い方をまとめてみます
AIDLの作成方法:AIDL構文は簡単で、1つ以上のメソッドを持つインタフェースを宣言したり、パラメータと戻り値を渡したりすることができます.これらのパラメータと戻り値は、リモートコールの必要性から任意のタイプではない.以下はAIDLがサポートするデータ型です:1.import宣言を必要としない簡単なJavaプログラミング言語タイプ(int,booleanなど)2.String,CharSequenceは特別な声明を必要としない.リスト、Map、Parcelablesのタイプである、これらのタイプに含まれるデータメンバーは、単純なデータ型、Stringなどの他の比でサポートされているタイプのみである.(また、Parcelablesを試していないので、Eclipse+ADTでコンパイルしていますが、後でサポートされるかもしれません).以下はAIDL構文://ファイル名:SomeClass.aidl//ファイルにはコメントがあります.javaと同じ//package以前のコメントは無視されます.//関数と変数の以前の注釈は、本番javaコードに追加されます.package com.cmcc.demo;//import導入文import com.cmcc.demo.ITaskCallback;interface ITaskBinder{//関数はjavaと同様に0から複数のパラメータがあり、1つの戻り値boolean isTaskRunning();void stopRunningTask();//パラメータは別のaidl定義インタフェースvoid registerCallback(ITaskCallback cb);void unregisterCallback(ITaskCallback cb);void unregisterCallback(ITaskCallback cb);//パラメータはString、in表入入力タイプ、outで出力タイプを表すことができる. int getCustomerList(in String branch, out String customerList); }
インタフェースを実現するにはいくつかの原則がある.投げ出された異常は呼び出し者に返さないでください.プロセス間放出異常処理は望ましくない. .IPC呼び出しは同期されます.IPCサービスが完了するのに数ミリ秒以上かかることを知っている場合は、Activityのプライマリ・スレッドで呼び出されるのを避ける必要があります.つまりIPC呼び出しはアプリケーションを保留することによってインタフェースの応答を失う.この場合、単一のスレッドで処理することを考慮すべきである. .AIDLインタフェースで静的属性を宣言することはできません.IPCの呼び出し手順:1.インタフェースタイプの変数を宣言します.aidlファイルで定義します. 2. ServiceConnectionを実装します. 3. ApplicationContextを呼び出す.bindService()は、ServiceConnection実装において伝達する. 4. サービス接続onServiceConnected()インプリメンテーションでは、IBinderインスタンス(呼び出されたService)が受信されます.YourInterfaceName.Stub.asInterface(IBinder)service)は、パラメータをYourInterfaceタイプに変換します. 5. インタフェースで定義されたメソッドを呼び出します.接続が切断されたときに投げ出されるDeadObjectExceptionの異常を常に検出します.リモートメソッドによってのみ放出されます. 6. 接続を切断し、インタフェースインスタンスのApplicationContextを呼び出す.unbindService()