資料來源:https://givemepass.blogspot.com/2015/11/aidl_27.html
為了預防哪天該網站的重要資料失聯,因此在這裡備份一下,以下為該網誌的內容:
如果想要透過另外一個Process幫你執行程式, 你可以利用AIDL去跟Process溝通,
在如何使用AIDL中有簡單介紹一下AIDL,
但是其實有很多細節並沒有說明, 所以這邊來實作非同步的範例。
更多的AIDL可以參考[官網]。
這個範例比較特別, 必須開兩個app才能達到IPC的效果。
首先第一個範例是遠端APP, 只需要建立AIDL檔案跟一支Service就好了。
- Remote Side
首先先建立AIDL檔案, 由於這個是非同步的方式,
因此我們要使用oneway關鍵字, 並且回傳值是void,
你會發現我們傳入的參數是一個callback物件,
所以下面還會在宣告一個aidl的callback interface。
package example.givemepass.aidlremotedemo; import example.givemepass.aidlremotedemo.IRemoteAIDLCallback; interface IRemoteAIDL { oneway void getRemoteName(IRemoteAIDLCallback callback); }
這支aidl是用來傳給client端的callback,
到時候Client端只要覆寫這個方法就可以收到msg了。
// IRemoteAIDLCallback.aidl package example.givemepass.aidlremotedemo; interface IRemoteAIDLCallback { void handleMsg(String name); }
再來就是Service部分, 這邊的Service是用來給Client呼叫,
因此使用我們的aidl檔案所產生的interface來實作成一個Binder物件,
透過service的onBind傳出去, 當結束的時候, 再指派成null即可。
那在我們的Service內有宣告一個flag,
是用來判斷client是否中斷聯繫, 如果有中斷聯繫,
則結束迴圈。
public class RemoteService extends Service { private boolean flag; private final IRemoteAIDL.Stub remoteBinder = new IRemoteAIDL.Stub(){ @Override public void getRemoteName(IRemoteAIDLCallback callback) throws RemoteException { while(!flag) { try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } callback.handleMsg("remote service"); } } }; @Override public IBinder onBind(Intent intent) { return remoteBinder; } @Override public boolean onUnbind(Intent intent) { flag = true; return super.onUnbind(intent); } }
Remote Side記得要開filter給系統辨識是要連結到這個Service。
menifest
<service
android:name=".RemoteService"
android:process=":remote">
<intent-filter>
<action android:name="service.remote" />
</intent-filter>
</service>
- Client Side
在這邊我們宣告了一個聯繫的Button跟一個中斷聯繫的Button,
TextView是用來顯示Remote端傳回來的訊息。
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="example.givemepass.aidlclientdemo.MainActivity">
<Button
android:text="connect remote"
android:id="@+id/connect_remote"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
<Button
android:layout_toRightOf="@id/connect_remote"
android:text="disconnect remote"
android:id="@+id/disconnect_remote"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
<TextView
android:layout_below="@id/connect_remote"
android:id="@+id/result"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!" />
</RelativeLayout>
這邊是我們的主程式
public class MainActivity extends AppCompatActivity { private IRemoteAIDL mService; private TextView result; private Button connectRemote; private Button disconnectRemote; private ServiceConnection mConnection = new ServiceConnection() { public void onServiceConnected(ComponentName className, IBinder service) { mService = IRemoteAIDL.Stub.asInterface(service); try { mService.getRemoteName(new IRemoteAIDLCallback.Stub(){ StringBuffer sb = new StringBuffer(); @Override public void handleMsg(final String name){ runOnUiThread(new Runnable() { @Override public void run() { sb.append(name + "\n"); result.setText(sb); } }); } }); } catch (RemoteException e) { e.printStackTrace(); } } public void onServiceDisconnected(ComponentName className) { mService = null; } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); result = (TextView) findViewById(R.id.result); connectRemote = (Button) findViewById(R.id.connect_remote); disconnectRemote = (Button) findViewById(R.id.disconnect_remote); connectRemote.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(); intent.setAction("service.remote"); intent.setPackage("example.givemepass.aidlremotedemo"); bindService(intent, mConnection, Context.BIND_AUTO_CREATE); } }); disconnectRemote.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { unbindService(mConnection); } }); } }
在如何使用bindService中,
有示範怎麼使用Service,
一開始我們就宣告了ServiceConnection物件,
比較不一樣的地方是我們拿到IBinder物件,
要透過我們所簽訂的AIDL轉成所對應的Binder物件,
mService = IRemoteAIDL.Stub.asInterface(service);
接著來實作我們的AIDL Callback方法。
mService.getRemoteName(new IRemoteAIDLCallback.Stub(){ StringBuffer sb = new StringBuffer(); @Override public void handleMsg(final String name){ runOnUiThread(new Runnable() { @Override public void run() { sb.append(name + "\n"); result.setText(sb); } }); } });
這樣一來就可以拿到在Remote Service所傳來的字串,
把它擷取起來裝到TextView內。
而當我們把聯繫的Button按下去以後,
就可以進行Service的連結。
Intent intent = new Intent(); intent.setAction("service.remote"); intent.setPackage("example.givemepass.aidlremotedemo"); bindService(intent, mConnection, Context.BIND_AUTO_CREATE);
這邊要明確的跟系統說我們要連到哪個package name以及它的filter字串是甚麼,
否則會找不到對應的Service。
一開始一定要將兩個apk都安裝到手機內, 否則會找不到Remote。
Place your comment