AIDL/IPC Android AIDL/IPC 進程通信機制——超詳細講解及用法案例剖析(播放器)

首先引申下AIDL,什麼是AIDL呢?IPC? ------ Designing a Remote Interface Using AIDL

通常情況下,我們在同一進程內會使用Binder、BroadCastReciver讓Service跟Activity進行通信,數據交互,數據共享,但是跨進程呢?

IPC:IPC接口,IPC接口本地代理  ------  Implementing IPC Using AIDL

AIDL意爲:Android Interface Define Language 即 Android 接口描述語言

與J2ee的區別:Java中不允許跨進程內存共享,只傳遞對象,採用RMI方式,也可以通過序列化傳遞對象

Android AIDL: 採用AIDL方式,可以傳遞Bundle,實現起來稍微麻煩些

實現過程:在服務端定義AIDL文件,ADT插件編譯器將其編譯之後,會在R文件同級目錄文件下生成.java文件,輕量級,使用代理類在客戶端和實現層間傳

遞值

語法:可以申明接口和方法,參數和返回值不是任何類型,不需要申明import,String,charSequence等

AIDL使用起來也不是很麻煩,我個人感覺很實用!所以今天就說說關於AIDL的特性,很多朋友在自己的博客只是說它的理論而不是寫一些實例來講解,我個人感覺你只是說說他的

理論,一般人根本就看不懂,而且本來就是很抽象的東西,又摸不着,你們寫了也是白寫!

完全不顧新手的感受,應該是寫一些小例子,讓大家參與進來一起學習和探討效果才最佳!而且也不是一大堆的文字,看起來也不會那麼乏味,大家說是不?

下面就開始今天的AIDL/IPC講解與學習吧~,概念都講的差不多了,接下來就是實戰,代碼小例子給初學者細嚼慢嚥

從服務端線程開始寫,結構圖:


EngineerJspRemoteService.aidl 寫完之後保存,就會在gen文件自動創建一個.java文件


EngineerJspService.java

[java] view plain copy
  1. package com.example.engineerjspserver;  
  2. /** 
  3.  * AIDL/IPC  
  4.  * @author Engineer-Jsp 
  5.  * @date 2014.11.17 
  6.  * */  
  7. import java.io.FileDescriptor;  
  8. import android.app.Service;  
  9. import android.content.Intent;  
  10. import android.media.MediaPlayer;  
  11. import android.os.IBinder;  
  12. import android.os.RemoteException;  
  13. import android.util.Log;  
  14. public class EngineerJspService extends Service{  
  15.     private static final String TAG = "EngineerJspService";  
  16.     MediaPlayer player;  
  17.     @Override  
  18.     public IBinder onBind(Intent intent) {  
  19.         Log.d(TAG, "EngineerJspService is onBind..."+"\n來自客戶端的綁定操作已經完成...");  
  20.         if(player==null){  
  21.             player = MediaPlayer.create(this, R.raw.music);  
  22.             player = new MediaPlayer();  
  23.             try {  
  24.                 FileDescriptor file = getResources().openRawResourceFd(R.raw.music).getFileDescriptor();  
  25.                 player.setDataSource(file);  
  26.                 player.setLooping(true);  
  27.             } catch (Exception e) {  
  28.                 Log.d(TAG, e.toString());  
  29.             }  
  30.             Log.d(TAG, "player is created..."+"\n服務端播放器已經實例化,準備就緒...");  
  31.         }  
  32.         return binder;  
  33.     }  
  34.     private IBinder binder = new EngineerJspRemoteService.Stub() {  
  35.           
  36.         @Override  
  37.         public void onStop() throws RemoteException {  
  38.             try {  
  39.                 if(player.isPlaying()){  
  40.                     Log.d(TAG, "客戶端進程對服務端進程執行了暫停操作...");  
  41.                     player.stop();  
  42.                 }  
  43.             } catch (Exception e) {  
  44.                 Log.d(TAG, e.toString());  
  45.             }  
  46.         }  
  47.           
  48.         @Override  
  49.         public void onPause() throws RemoteException {  
  50.             try {  
  51.                 if(player.isPlaying()){  
  52.                     return;  
  53.                 }  
  54.                 Log.d(TAG, "客戶端進程對服務端進程執行了開始操作...");  
  55.                 player.prepare();  
  56.                 player.start();  
  57.             } catch (Exception e) {  
  58.                 Log.d(TAG, e.toString());  
  59.             }  
  60.         }  
  61.     };  
  62.     public boolean onUnbind(Intent intent) {  
  63.         if(player!=null){  
  64.             player.release();  
  65.         }  
  66.         Log.d(TAG, "EngineerJspService is onUnBind..."+"\n客戶端已經解綁斷線,服務停止了...");  
  67.         return super.onUnbind(intent);  
  68.           
  69.     };  
  70.   
  71. }  
配置文件:

[html] view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     package="com.example.engineerjspserver"  
  4.     android:versionCode="1"  
  5.     android:versionName="1.0" >  
  6.   
  7.     <uses-sdk  
  8.         android:minSdkVersion="15"  
  9.         android:targetSdkVersion="18" />  
  10.   
  11.     <application  
  12.         android:allowBackup="true"  
  13.         android:icon="@drawable/ic_launcher"  
  14.         android:label="@string/app_name"  
  15.         android:theme="@style/AppTheme" >  
  16.               <activity  
  17.             android:name="com.example.engineerjspserver.MainAcivity"  
  18.             android:label="@string/app_name" >  
  19.             <intent-filter>  
  20.                 <action android:name="android.intent.action.MAIN" />  
  21.                 <category android:name="android.intent.category.LAUNCHER" />  
  22.             </intent-filter>  
  23.         </activity>  
  24.          <service android:name=".EngineerJspService" android:process=":remote">  
  25.             <intent-filter >  
  26.                 <action android:name="com.example.engineerjspserver.EngineerJspService"/>  
  27.             </intent-filter>  
  28.          </service>  
  29.     </application>  
  30.   
  31. </manifest>  

當完成上述操作之後,需要在res目錄下導入測試文件,比如我的測試文件一個music.mp3,需要與layout文件同級


搞定服務端之後,接下來就是客戶端線程編寫,之後進行進程通信測試

把服務端的aidl文件所在的包全部拷貝到客戶端的src下,把aidl文件除外的全部刪掉


EngineerJspActivity.java

[java] view plain copy
  1. package com.example.engineerjspclient;  
  2. /** 
  3.  * AIDL/IPC 
  4.  * @author Engineer-Jsp 
  5.  * @date 2014.11.17 
  6.  * */  
  7. import com.example.engineerjspserver.EngineerJspRemoteService;  
  8.   
  9. import android.os.Bundle;  
  10. import android.os.IBinder;  
  11. import android.util.Log;  
  12. import android.view.View;  
  13. import android.view.View.OnClickListener;  
  14. import android.widget.Button;  
  15. import android.widget.Toast;  
  16. import android.app.Activity;  
  17. import android.content.ComponentName;  
  18. import android.content.Intent;  
  19. import android.content.ServiceConnection;  
  20. public class EngineerJspActivity extends Activity {  
  21.     private static final String TAG = "EngineerJspActivity";  
  22.     private static final String ACTION = "com.example.engineerjspserver.EngineerJspService";  
  23.     private EngineerJspRemoteService EJService;  
  24.     private boolean isBind = false;  
  25.     private Button onPause,onStart;  
  26.       
  27.     private ServiceConnection connection = new ServiceConnection() {  
  28.         @Override  
  29.         public void onServiceDisconnected(ComponentName name) {  
  30.             isBind = false;  
  31.             EJService = null;  
  32.             Log.d(TAG, "onServiceDisconnected" +  
  33.                     "————"+"包名:"+name.getPackageName()+" "+"類名: "+name.getClassName()+"\n當客戶端線程斷線操作,服務將被清空...");  
  34.         }  
  35.         @Override  
  36.         public void onServiceConnected(ComponentName name, IBinder ibinder) {  
  37.             EJService = EngineerJspRemoteService.Stub.asInterface(ibinder);  
  38.             isBind = true;  
  39.             Log.d(TAG, "onServiceConnected————"+"包名:"+name.getPackageName()+" "+"類名: "+name.getClassName()+" "+"ibinder對象: "+ibinder.toString()+"\n客戶端線程與服務端線程連接中...");  
  40.         }  
  41.     };  
  42.     @Override  
  43.     protected void onCreate(Bundle savedInstanceState) {  
  44.         super.onCreate(savedInstanceState);  
  45.         setContentView(R.layout.activity_engineer_jsp);  
  46.         Log.d(TAG, "客戶端線程Activity創建成功,初始化控件與綁定服務中...");  
  47.         Bind();  
  48.         intiView();  
  49.     }  
  50.     @Override  
  51.     protected void onDestroy() {  
  52.         UnBind();  
  53.         Log.d(TAG, "客戶端線程被銷燬,活動被解綁停止了...");  
  54.         super.onDestroy();  
  55.     }  
  56.     public void Bind(){  
  57.         Log.d(TAG, "Activity已經創建完成,執行綁定服務端線程活動中...");  
  58.         Intent intent = new Intent(ACTION);  
  59.         bindService(intent, connection, BIND_AUTO_CREATE);  
  60.     }  
  61.     public void UnBind(){  
  62.         if(isBind){  
  63.             Log.d(TAG, "Activity被銷燬了,與服務端線程的通信將被解綁停止...");  
  64.             unbindService(connection);  
  65.             EJService = null;  
  66.             isBind = false;  
  67.         }  
  68.     }  
  69.     private void intiView(){  
  70.         onPause = (Button)findViewById(R.id.onPause);  
  71.         onStart = (Button)findViewById(R.id.onStop);  
  72.         onPause.setOnClickListener(listener);  
  73.         onStart.setOnClickListener(listener);  
  74.         Log.d(TAG, "服務端線程的組件初始化完畢...");  
  75.     }  
  76.     private OnClickListener listener = new OnClickListener(){  
  77.   
  78.         @Override  
  79.         public void onClick(View view) {  
  80.             if(view.getId()==R.id.onPause){  
  81.                 try {  
  82.                     EJService.onPause();  
  83.                     Log.d(TAG, "客戶端線程對服務端線程執行了播放操作...");  
  84.                     Toast.makeText(EngineerJspActivity.this,"客戶端線程對服務端線程執行了播放操作...", Toast.LENGTH_SHORT).show();  
  85.                 } catch (Exception e) {  
  86.                     Log.d(TAG, e.toString());  
  87.                 }  
  88.             }else{  
  89.                 try {  
  90.                     EJService.onStop();  
  91.                     Log.d(TAG, "客戶端線程對服務端線程執行了暫停操作...");  
  92.                     Toast.makeText(EngineerJspActivity.this,"客戶端線程對服務端線程執行了暫停操作...", Toast.LENGTH_SHORT).show();  
  93.                 } catch (Exception e) {  
  94.                     Log.d(TAG, e.toString());  
  95.                 }  
  96.             }  
  97.         }  
  98.           
  99.     };  
  100. }  

activity_engineer_jsp.xml

[html] view plain copy
  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:paddingBottom="@dimen/activity_vertical_margin"  
  6.     android:paddingLeft="@dimen/activity_horizontal_margin"  
  7.     android:paddingRight="@dimen/activity_horizontal_margin"  
  8.     android:paddingTop="@dimen/activity_vertical_margin"  
  9.     tools:context=".EngineerJspActivity" >  
  10.   
  11.     <Button  
  12.         android:id="@+id/onStop"  
  13.         android:layout_width="wrap_content"  
  14.         android:layout_height="wrap_content"  
  15.         android:layout_alignParentLeft="true"  
  16.         android:layout_alignParentRight="true"  
  17.         android:layout_below="@+id/button1"  
  18.         android:layout_marginTop="42dp"  
  19.         android:text="onStop" />  
  20.   
  21.     <Button  
  22.         android:id="@+id/onPause"  
  23.         android:layout_width="wrap_content"  
  24.         android:layout_height="wrap_content"  
  25.         android:layout_alignLeft="@+id/onStop"  
  26.         android:layout_alignParentRight="true"  
  27.         android:layout_below="@+id/onStop"  
  28.         android:text="onPause" />  
  29.   
  30. </RelativeLayout>  
配置文件:

[html] view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     package="com.example.engineerjspclient"  
  4.     android:versionCode="1"  
  5.     android:versionName="1.0" >  
  6.   
  7.     <uses-sdk  
  8.         android:minSdkVersion="15"  
  9.         android:targetSdkVersion="18" />  
  10.   
  11.     <application  
  12.         android:allowBackup="true"  
  13.         android:icon="@drawable/ic_launcher"  
  14.         android:label="@string/app_name"  
  15.         android:theme="@style/AppTheme" >  
  16.         <activity  
  17.             android:name="com.example.engineerjspclient.EngineerJspActivity"  
  18.             android:label="@string/app_name" >  
  19.             <intent-filter>  
  20.                 <action android:name="android.intent.action.MAIN" />  
  21.   
  22.                 <category android:name="android.intent.category.LAUNCHER" />  
  23.             </intent-filter>  
  24.         </activity>  
  25.     </application>  
  26.   
  27. </manifest>  

啓動服務端進程與客戶端進程(不在同一個進程內,也就是不在同一個APK程序之內),在DDMS裏進行進程和堆棧查看


當服務端進程開啓之後,會等待客戶端進程接入與操作,類似c/s模式,Socket,客戶端進程開啓之後,服務端測試數據如下


客戶端測試數據:


服務測試:

客戶端進程對服務端進程執行了播放操作:



客戶端進程對服務端進程執行了暫停操作:




兩個進程項目圖結構:


AIDL這個小實例內容就這麼多,概念也說得差不多了,你們好好看看,消化消化~

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章