成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術文章
文章詳情頁

詳解Android中Service AIDL的使用

瀏覽:5日期:2022-09-18 11:28:22
目錄前言Service基本用法——本地服務遠程服務 —— AIDL服務端客戶端前言

有些朋友可能是從事開發工作的時間不是特別的長,所以覺得Service相對與另外兩個組件activity、broadcast receiver來說,使用可能并不是特別的多,所以對Service來說,理解不是特別的深入,只是有一個大概的概念,今天就和一塊來走一下Service,希望能夠幫助到大家對Service有更深入的理解。

Service基本用法——本地服務

我們知道服務分為本地服務和遠程服務,而本地服務由于它的啟動方式不一樣,所以生命周期也就不一樣,對Service生命周期不熟悉的朋友,自行去百度一下啊。好了,那么我們分別看一下兩種不同的啟動方式。

我們先創建好Service:ServiceTest.java

package com.example.administrator.servicetestaidl;import android.app.Service;import android.content.Intent;import android.os.IBinder;import android.support.annotation.IntDef;import android.util.Log;public class ServiceTest extends Service { @Override public void onCreate() {super.onCreate();Log.d('ServiceTest',' -----> onCreate'); } @Override public int onStartCommand(Intent intent,int flags, int startId) {Log.d('ServiceTest',' -----> onStartCommand');return super.onStartCommand(intent, flags, startId); } @Override public void onDestroy() {super.onDestroy();Log.d('ServiceTest',' -----> onDestroy'); } @Override public IBinder onBind(Intent intent) {// TODO: Return the communication channel to the service.throw new UnsupportedOperationException('Not yet implemented'); }}

在看看MainActivity的代碼:

package com.example.administrator.servicetestaidl;import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.view.View;import android.widget.Button;public class MainActivity extends Activity { private Button startService, stopService; @Override protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);startService = (Button) findViewById(R.id.start_service);stopService = (Button) findViewById(R.id.stop_service);/** * 開啟服務 */startService.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {Intent startService = new Intent(MainActivity.this,ServiceTest.class);startService(startService); }});/** * 停止服務 */stopService.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {Intent stopService = new Intent(MainActivity.this,ServiceTest.class);stopService(stopService); }}); }}

布局activity_main

<?xml version='1.0' encoding='utf-8'?><LinearLayout xmlns:android='http://schemas.android.com/apk/res/android' android:layout_width='match_parent' android:layout_height='match_parent' android:orientation='vertical'> <Buttonandroid: android:layout_width='wrap_content'android:layout_height='wrap_content'android:text='開啟服務' /> <Buttonandroid: android:layout_width='wrap_content'android:layout_height='wrap_content'android:text='停止服務' /></LinearLayout>

配置文件AndroidManifest.xml

<?xml version='1.0' encoding='utf-8'?><manifest xmlns:android='http://schemas.android.com/apk/res/android' package='com.example.administrator.servicetestaidl'> <applicationandroid:allowBackup='true'android:icon='@mipmap/ic_launcher'android:label='@string/app_name'android:roundIcon='@mipmap/ic_launcher_round'android:supportsRtl='true'android:theme='@style/AppTheme'><activity android:name='.MainActivity'> <intent-filter><action android:name='android.intent.action.MAIN' /><category android:name='android.intent.category.LAUNCHER' /> </intent-filter></activity><service android:name='.ServiceTest' android:enabled='true' android:exported='true'></service> </application></manifest>

上面的代碼很簡單,并不難理解,在頁面上加兩個按鈕,一個是啟動服務,一個是銷毀服務的,并且我們在ServiceTest里面的幾個方法都加上了log,那我們點擊開啟服務,看看Log,如圖:

詳解Android中Service AIDL的使用

然后我們多次點擊開啟服務,如圖:

詳解Android中Service AIDL的使用

我們看到,后面即使多點幾下這個開啟服務,但是也只會調onStartCommand方法,onCreate方法并不會重復調用,那是因為我們點擊Service,由于該service已經存在,所以并不會重新創建,所以onCreate方法只會調用一次。我們還可以到手機的應用程序管理界面來檢查一下Service是不是正在運行,如下圖所示:

詳解Android中Service AIDL的使用

那當我們點擊停止服務按鈕呢,看看log:如圖

詳解Android中Service AIDL的使用

這時候說明了服務已經銷毀了。

有些朋友可能注意到了,我們剛剛那種啟動服務的方式,好像除了對Service進行開啟和銷毀以外,很難在activity里進行對Service進行控制,什么意思呢?舉個例子,如果說我現在用Service進行下載某些東西,我現在在Service寫有下載這兩個東西的方法,方法a,方法b,那么我怎樣在activity里面控制什么時候調用方法a,什么時候調用方法b呢,如果按照原本的啟動方式,好像并不好實現,或者說靈活性很差,那么有沒有辦法辦到呢,接著看Service另一種啟動方式。在前面我們有一個方法一直都沒有動onBind方法,我們就從這個方法入手,先看ServiceTest代碼:

package com.example.administrator.servicetestaidl;import android.app.Service;import android.content.Intent;import android.os.Binder;import android.os.IBinder;import android.support.annotation.IntDef;import android.util.Log;public class ServiceTest extends Service { @Override public void onCreate() {super.onCreate();Log.d('ServiceTest',' -----> onCreate'); } @Override public int onStartCommand(Intent intent,int flags, int startId) {Log.d('ServiceTest',' -----> onStartCommand');return super.onStartCommand(intent, flags, startId); } @Override public void onDestroy() {super.onDestroy();Log.d('ServiceTest',' -----> onDestroy'); } @Override public IBinder onBind(Intent intent) {return new Mybind(); } class Mybind extends Binder{public void getString(){ Log.d('ServiceTest',' -----> getString');} }}

在ServiceTest中增加了一個內部類Mybind,并且在Mybind中增加一個getString方法,在方法中打印log,然后在onBind方法中返回Mybind對象。

再看看MainActivity的代碼

package com.example.administrator.servicetestaidl;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.view.View;import android.widget.Button;public class MainActivity extends Activity { private Button startService,stopService,bindService,unbindService; private ServiceTest.Mybind mybind; private ServiceConnection connection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) { mybind = (ServiceTest.Mybind) service; mybind.getString(); //獲取到getString方法}@Overridepublic void onServiceDisconnected(ComponentName name) {} }; @Override protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);startService = (Button) findViewById(R.id.start_service);stopService = (Button) findViewById(R.id.stop_service);bindService = (Button) findViewById(R.id.bind_service);unbindService = (Button) findViewById(R.id.unbind_service);/** * 開啟服務 */startService.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {Intent startService = new Intent(MainActivity.this,ServiceTest.class);startService(startService); }});/** * 停止服務 */stopService.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {Intent stopService = new Intent(MainActivity.this,ServiceTest.class);stopService(stopService); }});/** * 綁定服務 */bindService.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {Intent bindService = new Intent(MainActivity.this,ServiceTest.class);bindService(bindService,connection,BIND_AUTO_CREATE); }});/** * 解綁服務 */unbindService.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {unbindService(connection); }}); }}

主頁面布局:activity_main

<?xml version='1.0' encoding='utf-8'?><LinearLayout xmlns:android='http://schemas.android.com/apk/res/android' android:layout_width='match_parent' android:layout_height='match_parent' android:orientation='vertical'> <Buttonandroid: android:layout_width='wrap_content'android:layout_height='wrap_content'android:text='開啟服務' /> <Buttonandroid: android:layout_width='wrap_content'android:layout_height='wrap_content'android:text='停止服務' /> <Buttonandroid: android:layout_width='wrap_content'android:layout_height='wrap_content'android:text='綁定服務' /> <Buttonandroid: android:layout_width='wrap_content'android:layout_height='wrap_content'android:text='解綁服務' /></LinearLayout>

可以看到,這里我們首先創建了一個ServiceConnection的匿名類,在里面重寫了onServiceConnected()方法和onServiceDisconnected()方法,這兩個方法分別會在Activity與Service建立關聯和解除關聯的時候調用。在onServiceConnected()方法中,我們又通過向下轉型得到了MyBind的實例,有了這個實例,Activity和Service之間的關系就變得非常緊密了?,F在我們可以在Activity中根據具體的場景來調用MyBind中的任何public方法,即實現了Activity指揮Service干什么Service就去干什么的功能。

當我們點擊綁定服務的時候,結果如下,如圖

詳解Android中Service AIDL的使用

點擊解綁服務的時候,結果如下,如圖

詳解Android中Service AIDL的使用

注意:Service 是運行在后臺,沒有可視化的頁面,我們很多時候會把耗時的操作放在Service中執行,但是注意,Service是運行在主線程的,不是在子線程中,Service和Thread沒有半毛錢的關系,所以如果在Service中執行耗時操作,一樣是需要開起線程,否則會引起ANR,這個需要區別開來。

遠程服務 —— AIDL

AIDL(Android Interface Definition Language)是Android接口定義語言的意思,它可以用于讓某個Service與多個應用程序組件之間進行跨進程通信,從而可以實現多個應用程序共享同一個Service的功能。實際上實現跨進程之間通信的有很多,比如廣播,Content Provider,但是AIDL的優勢在于速度快(系統底層直接是共享內存),性能穩,效率高,一般進程間通信就用它。

既然是跨進程,那必須的有兩個應用,一個是service端,一個是client端,然后實現客戶端從服務端獲取數據。那么我們創建一個服務端,項目結構如圖所示:

詳解Android中Service AIDL的使用

服務端

我們在服務端下建立一個MyAIDLService.aidl文件,目錄結構為如圖所示:

詳解Android中Service AIDL的使用

然后,我們在MyAIDLService下增加一個獲取字符串的方法。代碼如下:(注:剛剛建立的aidl文件中存在一個方法,那個方法可以忽略,可以刪掉不要)

// MyAIDLService.aidlpackage aidl;// Declare any non-default types here with import statementsinterface MyAIDLService { //獲取String數據 String getString();}

創建完aidl文件以后,我們build一下項目,然后會在build - >generated ->source ->aidl->debug下會生成一個aidl文件,那說明AIDL文件已經編譯成功。

接著建立一個MyService類,代碼如下:

package com.example.service;import android.app.Service;import android.content.Intent;import android.os.Binder;import android.os.IBinder;import android.os.RemoteException;import android.util.Log;import java.util.Map;import aidl.MyAIDLService;public class MyService extends Service { @Override public void onCreate() {super.onCreate(); } @Override public int onStartCommand(Intent intent, int flags, int startId) {return super.onStartCommand(intent, flags, startId); } @Override public void onDestroy() {super.onDestroy(); } @Override public IBinder onBind(Intent intent) {return new Mybind(); } class Mybind extends MyAIDLService.Stub {@Overridepublic String getString() throws RemoteException { String string = '我是從服務起返回的'; return string;} }}

代碼看起來是不是很熟悉,唯一不一樣的就是原來在本地服務的時候內部類繼承的是Binder,而現在繼承的是MyAIDLService.Stub,繼承的是我們剛剛建立的aidl文件,然后實現我們剛剛的定義的getString()方法,在這里,我們只是返回一句話,'我是從服務起返回的'~~~~~~~~~~~

客戶端

首先將剛剛在服務端創建的MyAIDLService原封不動的復制到客戶端來。(注意:路徑要一模一樣)。接著我們在客戶端的MainActivity中加兩個按鈕,并且和服務端進行相連,代碼如下:

MainActivity

package com.example.administrator.servicetestaidl;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.view.View;import android.widget.Button;import android.widget.TextView;import aidl.MyAIDLService;public class MainActivity extends Activity { private Button bindService,unbindService; private TextView tvData; private MyAIDLService myAIDLService; private ServiceConnection connection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) { myAIDLService = MyAIDLService.Stub.asInterface(service); try {String str = myAIDLService.getString();tvData.setText(str); } catch (RemoteException e) {e.printStackTrace(); }}@Overridepublic void onServiceDisconnected(ComponentName name) { myAIDLService = null;} }; @Override protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);bindService = (Button) findViewById(R.id.bind_service);unbindService = (Button) findViewById(R.id.unbind_service);tvData = (TextView) findViewById(R.id.tv_data);/** * 綁定服務 */bindService.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {Intent intent = new Intent();intent.setAction('com.example.service.MyService');//從 Android 5.0開始 隱式Intent綁定服務的方式已不能使用,所以這里需要設置Service所在服務端的包名intent.setPackage('com.example.service');bindService(intent, connection, BIND_AUTO_CREATE); }});/** * 解綁服務 */unbindService.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) {unbindService(connection); }}); }}

大家是不是感覺和連接本地服務的代碼差不多,沒錯,這里只需要注意兩個地方,一個是綁定服務的時候,因為從 Android 5.0開始 隱式Intent綁定服務的方式已不能使用,所以這里需要設置Service所在服務端的包名

那么這個action是怎么來的呢,我們回來服務端的AndroidManifest.xml,代碼如下

<?xml version='1.0' encoding='utf-8'?><manifest xmlns:android='http://schemas.android.com/apk/res/android' package='com.example.service'> <applicationandroid:allowBackup='true'android:icon='@mipmap/ic_launcher'android:label='@string/app_name'android:roundIcon='@mipmap/ic_launcher_round'android:supportsRtl='true'android:theme='@style/AppTheme'><activity android:name='.MainActivity'> <intent-filter><action android:name='android.intent.action.MAIN' /><category android:name='android.intent.category.LAUNCHER' /> </intent-filter></activity><service android:name='.MyService' > <intent-filter><action android:name='com.example.service.MyService' /> </intent-filter></service> </application></manifest>

另一個需要注意的就是獲取MyAIDLService對象是通過MyAIDLService.Stub.asInterface(service);這里大家需要注意一下的。不過還有一點需要說明的是,由于這是在不同的進程之間傳遞數據,Android對這類數據的格式支持是非常有限的,基本上只能傳遞Java的基本數據類型、字符串、List或Map等。那么如果我想傳遞一個自定義的類該怎么辦呢?這就必須要讓這個類去實現Parcelable接口,并且要給這個類也定義一個同名的AIDL文件。這部分內容并不復雜,而且和Service關系不大,所以就不再詳細進行講解了,感興趣的朋友可以自己去查閱一下相關的資料。

注意:從服務器復制過來的aidl文件不能直接放到Java文件夾下面,必須建立一個aidl文件夾存放,否則會編譯不成功

好了,到這里,基本上就結束了,附上一張效果圖:

詳解Android中Service AIDL的使用

最后附上源碼鏈接

本地服務源碼:https://github.com/343661629/nativeService

遠程服務源碼:https://github.com/343661629/remoteService

以上就是詳解Android中Service AIDL的使用的詳細內容,更多關于Android中Service AIDL的使用的資料請關注好吧啦網其它相關文章!

標簽: Android
相關文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
成人一区二区三区中文字幕| 免费美女久久99| 亚洲国产精品久久艾草纯爱| 欧美国产免费| 日韩你懂的在线播放| 国内精品视频一区二区三区八戒| 久久久夜精品| 亚洲午夜久久久久久久久电影院| 亚洲国产日韩欧美| 久久久777精品电影网影网| 国产成人精品www牛牛影视| 欧美色图天堂网| 亚洲小说春色综合另类电影| 最新日韩在线| 亚洲日本在线视频观看| 亚洲欧洲日本一区二区三区| 亚洲欧洲精品一区二区三区 | 免费在线亚洲| 亚洲一区二区美女| 香蕉久久夜色精品| 亚洲午夜在线电影| 久久动漫亚洲| 日韩成人免费看| 老司机精品视频网站| 亚洲高清不卡在线| 久久久久网址| 麻豆一区二区三区| 欧美乱妇15p| 成人午夜精品在线| 久久色.com| 欧美不卡高清| 中文字幕一区二区三区av| 亚洲日本欧美在线| 亚洲成av人片一区二区| 在线观看成人免费视频| 黄色小说综合网站| 欧美疯狂性受xxxxx喷水图片| 国产乱码字幕精品高清av| 日韩一区二区中文字幕| 丁香另类激情小说| 久久久噜噜噜久久人人看| 99久久综合精品| 中文字幕va一区二区三区| 亚洲日韩成人| 视频在线在亚洲| 在线播放欧美女士性生活| 国产jizzjizz一区二区| 久久久久久久久久久久久女国产乱| 欧美1级日本1级| 国产精品久久久久久久久动漫 | 国产精品一二| 日本不卡视频一二三区| 欧美视频精品在线| 91丨国产丨九色丨pron| 一区二区理论电影在线观看| 久久国产一区| 国产精品白丝jk白祙喷水网站| 国产亚洲一区二区三区| 日韩午夜精品| 久久国产日韩欧美精品| 精品粉嫩超白一线天av| 国产精品豆花视频| 亚洲愉拍自拍另类高清精品| 欧美自拍偷拍一区| 91视频免费看| 亚洲综合成人在线| 欧美日韩激情在线| jizzjizzjizz欧美| 亚洲美女区一区| 欧洲色大大久久| 波多野洁衣一区| 亚洲男帅同性gay1069| 色婷婷av一区二区三区之一色屋| 国产黄色91视频| 国产喷白浆一区二区三区| 国产精品免费看| 国产一区二区三区在线观看精品 | 91免费观看在线| 亚洲激情中文1区| 欧美日韩国产精选| 欧美日本亚洲| 美洲天堂一区二卡三卡四卡视频| 久久免费电影网| 亚洲一区二区在线看| 国产成人在线视频网站| 亚洲色欲色欲www| 欧美三级中文字| 欧美亚韩一区| 日韩激情中文字幕| 久久久.com| 色天使色偷偷av一区二区| 成人精品亚洲人成在线| 亚洲精品中文字幕乱码三区| 欧美肥妇bbw| 狠狠爱成人网| 国产一区二区视频在线播放| 亚洲激情五月婷婷| 日韩一区二区电影网| 国产一区日韩一区| 精品一区二区三区免费观看| 国产精品国产成人国产三级| 91福利精品第一导航| 欧美另类高清视频在线| 日韩精品一区第一页| 久久精品人人做人人综合| 在线欧美日韩精品| 国产精品v欧美精品v日韩| 老司机一区二区| 自拍偷拍国产精品| 日韩三级视频中文字幕| 国产乱人伦精品一区二区| 成人a级免费电影| 亚洲成人先锋电影| 国产欧美精品区一区二区三区| 在线观看免费一区| 国产日韩一区| 欧美视频不卡| 成人不卡免费av| 免费精品视频在线| 亚洲综合色噜噜狠狠| 久久精品夜色噜噜亚洲a∨| 在线免费观看日韩欧美| 91久久精品www人人做人人爽| 成人av电影在线| 麻豆精品一二三| 亚洲综合成人在线| 国产人成一区二区三区影院| 欧美一区二区三区白人| 久久精品国产第一区二区三区最新章节 | 亚洲激情网站免费观看| 欧美精品一区二区三区四区| 欧美日韩一级黄| 久久成人精品| 一色屋精品视频在线看| 9久草视频在线视频精品| 免费观看在线色综合| 日韩一区中文字幕| 久久影院视频免费| 欧美日韩中文另类| 久久久www| 亚洲国产精品日韩| 99精品国产99久久久久久白柏| 麻豆精品在线视频| 亚洲成av人**亚洲成av**| 成人免费视频在线观看| 欧美变态tickling挠脚心| 欧美制服丝袜第一页| 亚洲一区二区三区免费观看| 国产精品v一区二区三区| jlzzjlzz亚洲女人18| 久久黄色级2电影| 日韩—二三区免费观看av| 亚洲精品久久久蜜桃| 国产精品每日更新在线播放网址| 日韩午夜在线观看| 欧美群妇大交群的观看方式| 色婷婷国产精品| 久久一区欧美| 国产亚洲激情| 日韩一区二区久久| 欧美日韩一区综合| 91免费观看在线| 成人性生交大片免费看中文网站| 国产在线一区观看| 日本欧美肥老太交大片| 亚洲另类在线一区| 亚洲日本欧美天堂| 亚洲老妇xxxxxx| 一区二区三区精品| 亚洲一区在线观看免费观看电影高清 | www.亚洲激情.com| 国产91富婆露脸刺激对白| 国产另类ts人妖一区二区| 麻豆高清免费国产一区| 日韩中文字幕不卡| 日本三级韩国三级欧美三级| 午夜精品一区在线观看| 亚洲在线视频免费观看| 夜夜亚洲天天久久| 亚洲午夜免费电影| 午夜成人免费视频| 婷婷久久综合九色综合伊人色| 亚洲一区二区三区中文字幕| 亚洲国产精品麻豆| 日韩中文字幕区一区有砖一区| 视频一区视频二区在线观看| 日产精品久久久久久久性色| 日本不卡在线视频| 久久99日本精品| 免费日韩伦理电影| 精品一二线国产| 国产精品资源在线| 99久久99久久免费精品蜜臀| 欧美凹凸一区二区三区视频| 亚洲私人影院| 一级日韩一区在线观看| 亚洲欧美卡通另类91av| 久久亚洲影院| 欧美日韩亚州综合| 日韩欧美国产系列|