Android通過AIDL在兩個APP之間Service通信
一、項目介紹
【知識準(zhǔn)備】
?、貯ndroid Interface definition language(aidl,android接口定義語言),其目的實現(xiàn)跨進(jìn)程的調(diào)用。進(jìn)程是程序在os中執(zhí)行的載體,一個程序?qū)?yīng)一個進(jìn)程,不同進(jìn)程就是指不同程序,aidl實現(xiàn)不同程序之間的調(diào)用。
②主線程與子線程通信使用handler,handler可以在子線程中發(fā)出消息,在主線程處理消息,從而完成線程之間的通信,即使有多個線程,仍然是一個程序。
?、鄄煌绦蛑g需要通過aidl通信,通信方式可以有多種,aidl是其中一種。實現(xiàn)的結(jié)果就像自己的程序調(diào)用自己的其他方法一樣,感覺就像一個程序。
?、軜I(yè)務(wù)場景:例如購物app需要支付,購物app是淘寶,支付app是支付寶。所以就需要不同的程序進(jìn)行通信。
二、首先介紹一個App之間的Service和Activity之間的通信
【項目結(jié)構(gòu)】
【MyService】
【提示】
①創(chuàng)建Service
?、谌绻皇峭ㄟ^上述方法創(chuàng)建,一定要記得注冊
<service android:name=".MyService" android:enabled="true" android:exported="true"> </service>
【代碼】
public class MyService extends Service { public MyService() { } @Override public IBinder onBind(Intent intent) { return new MyBinder();//return MyBinder通過ServiceConnection在activity中拿到MyBinder } @Override public int onStartCommand(Intent intent, int flags, int startId) { return super.onStartCommand(intent, flags, startId); } public void payService(){ Log.i("MyService", "payService: --------"); } class MyBinder extends Binder{ public void pay(){ payService(); }//通過Binder實例將service中的方法暴露出去 } }
【layout_main】
添加按鈕,點擊便于調(diào)用
<Button android:id="@+id/btn_paly" android:text="Pay" android:layout_width="wrap_content" android:layout_height="wrap_content" />
【MainActivity】
public class MainActivity extends AppCompatActivity { MyService.MyBinder binder = null; ServiceConnection conn; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Button btnPlay = (Button) findViewById(R.id.btn_paly); conn = new ServiceConnection() { @Override public void onServiceConnected(ComponentName componentName, IBinder iBinder) { binder = (MyService.MyBinder) iBinder; } @Override public void onServiceDisconnected(ComponentName componentName) { } }; Intent intent = new Intent(MainActivity.this,MyService.class); bindService(intent,conn,BIND_AUTO_CREATE);//開啟服務(wù) btnPlay.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { if (binder!=null){ binder.play(); } } }); } }
【效果】
點擊后輸出service中pay方法中的內(nèi)容
三、兩個App之間的Service通信
【項目結(jié)構(gòu)】
【步驟】
①在AppPayProvider中創(chuàng)建MyService
代碼同上
【注冊】
Ⅰ、注冊時(android:enabled="true" android:exported="true" )設(shè)置為true,將Service暴露出去,另一個App才能訪問到它
?、?、添加『 <intent-filter> 』。由于不是同一個App,通過intent-filter對Intent進(jìn)行過濾,讓另一個app通過action開啟服務(wù)
<service android:name=".MyService" android:enabled="true" android:exported="true"> <!--enable:ture設(shè)置可用 exported:ture對外暴露 --> <intent-filter> <action android:name="com.xqz.apppayprovider.MyService" /> </intent-filter> </service>
②MainActivity和layout_main保留創(chuàng)建時不作任何修改,但也不要刪掉,因為安裝程序必須提供起始頁面,否則將會出錯
③在AppPayProvider中添加AIDL
【代碼】
【提示】接口中定義中方法要和Service中的MyBinder中的方法一致
④再創(chuàng)建好AIDL,添加完方法后,android studio需要對這個aidl進(jìn)行編譯,會自動按aidl規(guī)范生成一個Binder子類的代碼。
⑤對MyService中的MyBinder進(jìn)行修改
【提示】繼承IPay.Stub。在這之前必須Make Project,否則將沒有只能聯(lián)想
⑥創(chuàng)建AppPayUser對AppPayProvider中的MyService進(jìn)行操作
【layout-main】
<Button android:id="@+id/btnPay" android:text="pay" android:layout_width="wrap_content" android:layout_height="wrap_content" />
⑦將AppPayProvider中AIDL拷貝到AppPayUser中
【提示】Ⅰ、包名要相同,按目錄位置復(fù)制,通過下述方法,直接在文件夾進(jìn)行復(fù)制。『此處可以查看項目結(jié)構(gòu),可以看到包名是相同的』
Ⅱ、同樣拷貝過來后需要Make Project
⑧【AppPayUser-MainActivity】
public class MainActivity extends AppCompatActivity { Button btnPay; private IPay myBinder;//定義AIDL ServiceConnection conn = new ServiceConnection() { @Override public void onServiceConnected(ComponentName componentName, IBinder iBinder) { myBinder = IPay.Stub.asInterface(iBinder); } @Override public void onServiceDisconnected(ComponentName componentName) { } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Intent intent = new Intent(); intent.setAction("com.xqz.apppayprovider.MyService"); //表示按照什么進(jìn)行過濾,啟動意圖 /*android5.0之后,如果servicer不在同一個App的包中, 需要設(shè)置service所在程序的包名 (包名可以到App的清單文件AndroidManifest中查看)*/ intent.setPackage("com.xqz.apppayprovider"); bindService(intent,conn,BIND_AUTO_CREATE);//開啟Service btnPay = (Button) findViewById(R.id.btnPay); btnPay.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { try { myBinder.pay(); } catch (RemoteException e) { //因為是跨程序調(diào)用服務(wù),可能會出現(xiàn)遠(yuǎn)程異常 e.printStackTrace(); } } }); } }
【安裝】
先安裝AppPayProvider再安裝AppPayUser。
【效果】
將run中的 視圖調(diào)到AppPayProvider,點擊模擬器AppPayUser中的pay按鈕,將會執(zhí)行AppPayProvider中MyService中pay方法中的內(nèi)容。
四、總結(jié)
【跨App和同App之間的區(qū)別】
①跨App開啟服務(wù)是提供服務(wù)的App需要設(shè)置intent-filter過濾器,控制服務(wù)的App需要通過。setAction和setPackage方法進(jìn)行設(shè)置action和包名,才能開啟服務(wù)。而同App只需要指定啟動的service就可。
②跨App的MyBinder實例要通過AIDL獲取,兩個應(yīng)用定義同樣的接口的方法,通過對應(yīng)的AIDL名稱.Stub.asInterface方法得到binder實例,然后就和同App的myBinder使用么有區(qū)別了。
③跨App的MyBinder對象的使用必須捕獲異常,而同App不需要。
④可以根據(jù)上方簡單的例子實現(xiàn)很多類似的功能。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android如何讓APP無法在指定的系統(tǒng)版本上運(yùn)行(實現(xiàn)方法)
這篇文章主要介紹了Android如何讓APP無法在指定的系統(tǒng)版本上運(yùn)行(實現(xiàn)方法),本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-02-02Android編程實現(xiàn)ListView頭部ViewPager廣告輪詢圖效果
這篇文章主要介紹了Android編程實現(xiàn)ListView頭部ViewPager廣告輪詢圖效果,較為詳細(xì)的分析了自定義ListView實現(xiàn)ViewPager廣告圖輪詢的相關(guān)技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-10-10解決Android平臺中應(yīng)用程序OOM異常的方法
這篇文章主要介紹了解決Android平臺中應(yīng)用程序OOM異常的方法,通常這一塊也是程序中的重點之一,感興趣的小伙伴們可以參考一下2015-12-12Android WebViewClient 的 `shouldOverrideUrlLoa
這篇文章主要介紹了Android WebViewClient 的 shouldOverrideUrlLoading方法,了解并正確實現(xiàn) WebViewClient 中的 shouldOverrideUrlLoading 方法對于在你的 Android 應(yīng)用中提供順暢且安全的瀏覽體驗至關(guān)重要,需要的朋友可以參考下2024-07-07android中Intent傳值與Bundle傳值的區(qū)別詳解
本篇文章是對android中Intent傳值與Bundle傳值的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05