Service兩種啟動方式
startService
啟動流程
oncreate->onStartCommand->onStart
-
多次啟動
onstartCommand->onStart
- bindService
oncreate->onBind
service中不能做耗時任務(不能超過5s),但是service可以通過新開一個線程去做耗時任務,完成后告訴頁面做完(為什么不在actiivty中直接做呢?因為activity是很容易被銷毀的,用戶按下返回鍵?用戶推出系統?等等,這樣一旦被銷毀了,我們就無法獲取到新開的那個線程,也無法控制,但是service不一樣,service是運行在后臺,即使頁面銷毀了也可以找到service,通過bindservice方法,然后在servcerconnection方法中獲取到IBinder,從而對service進行操作)
這也就衍生了跨進程服務aidl
aidl跨進程
(通過一個service的操作把各個進程鏈接起來,service就是這個節點)
- aidl 初學指南
- 明確你要學aidl干嘛?
- aidl能干嘛
- aidl怎么干的
大家學aidl 是為了跨進程通信,但是一個app為什么要跨進程呢?因為android 每個app就是運行在一個獨立的進程中,一旦進程中有些邏輯沒有處理好,或者是占用內存過大,引發了oom,crash會導致當前app直接閃退,嚴重影響了用戶體驗,所以我們就要把一些不是那么重要的任務處理或者需要比較大的內存的任務,或者是容易引發oom(webview內存泄漏)放在新的進程中,這樣 就可以避免主app閃退,但是這樣就有兩個進程,進程間通信那么就用到了aidl
aidl用于解決進程通信。
aidl原理(利用binder機制(書上說的) 我理解的表層意思,利用service實現(service可以脫離組件單獨存活,所以擁有比組件更長的生命周期))
實現一個彈幕接收遠程service
首先注冊一個service并注明在新的線程中
然后通過service的onBind方法返回aidl的對象,aidl中實現具體的方法,在組件中實現serviceconnection中實現了可以調用到aidl中方法 這樣我們就實現了當前進程調用另一個進程中的方法,實現了跨進程通信
接下來我們具體說一下aidl(搭建環境是android studio2.3.3)
1.首先是明確aidl能傳遞什么類型的字段,總體來說能傳遞java原生的類型,就像是什么int,long,float,list,map等等都可以,但是如果要是想傳遞自己定一個的類型,那么該類型必須實現Parcelable接口(這是一個android提供序列化的方法)
2.創建aidl文件 里面定義你要實現的方法(在android studio中 新建的方法如圖
)
(不帶自己創建的類的aidl)新建完成后 就把里面自帶的那個方法刪掉(不刪也行,個人習慣),然后寫出你想要實現的方法,這樣就可以了(如圖
)
其中connect,plus這兩個方法就是完成原生類型的aidl方法,
如果想要實現參數是自己的類,那么首先這個類在你自己的app中要實現Parcelable方法這個還不夠,還要創建一個相應的aidl文件,但是創建發現,android studio會提示重名了?(這個你不用管,隨便弄一個名字創建一個aidl,然后把里面的東西刪掉,改成如圖
這樣,這里 Message改成你自己想要的類名就行)
然后就是開始創建自己的aidl實例
public class LibHandler extends IHandler.Stub {
private RemoteCallbackList<ICallback> iCallbackRemoteCallbackList;
private Handler handler = new Handler();
public LibHandler() {
iCallbackRemoteCallbackList = new RemoteCallbackList<>();
}
@Override
public void connect() throws RemoteException {
Log.d("aidl", "connect()");
}
@Override
public void sendMessage(Message message) throws RemoteException {
Log.d("aidl", message.content);
}
@Override
public int plus(int a, int b) throws RemoteException {
Log.d("aidl", a + " " + b);
pushMessage();
return a + b;
}
int i = 1;
private void pushMessage() {
i++;
handler.postDelayed(new Runnable() {
@Override
public void run() {
try {
int length = iCallbackRemoteCallbackList.beginBroadcast();
Log.d("aidl", " " + length);
for (int i = 0; i < length; i++) {
iCallbackRemoteCallbackList.getBroadcastItem(i).callback(new Random().nextInt(100));
}
} catch (RemoteException e) {
e.printStackTrace();
}
iCallbackRemoteCallbackList.finishBroadcast();
pushMessage();
}
}, 2000);
}
@Override
public void register(final ICallback callback) throws RemoteException {
if (callback != null) {
iCallbackRemoteCallbackList.register(callback);
}
}
@Override
public void unregister(ICallback callback) throws RemoteException {
if (callback != null) {
iCallbackRemoteCallbackList.unregister(callback);
}
}
@Override
public IBinder asBinder() {
Log.d("aidl", "asBinder");
return super.asBinder();
}
}
就像這樣就行了,
如果期間出現一些什么找不到類的報錯 build-》make project一下,android studio 會生成aidl對應的java文件,這樣我們繼承的aidl文件就會被as找到
然后在service的onBind方法中返回我們的aidl 就可以了(如圖
)
這樣 我們的service就徹底變成了遠程的service
接下來看看組件中 是怎么實現與遠程service鏈接的
Intent intent = new Intent();
intent.setAction("com.boger.aidl.service");
intent.setPackage("com.boger.aidldemo");
bindService(intent, connection, Context.BIND_AUTO_CREATE);
ServiceConnection connection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
mServer = IHandler.Stub.asInterface(service);
try {
// String content = null;
mServer.connect();
int value = mServer.plus(3, 5);
Message message = new Message();
message.setContent("content");
message.setId(111);
mServer.sendMessage(message);
mServer.register(callback);
tv.setText(value + "");
} catch (RemoteException e) {
e.printStackTrace();
}
}
@Override
public void onServiceDisconnected(ComponentName name) {
mServer = null;
}
};
ICallback callback = new ICallback.Stub() {
@Override
public void callback(int id) throws RemoteException {
handler.sendEmptyMessage(id);
}
};
就這樣 就可以了
當然我這是一種隱式調用(針對別的app啟動該service,自己的app可以直接Intent intent = new Intent(package, Servicename.class)),在5.0以上service禁止隱式調用,會報一個 Service Intent must be explicit,去contextImpl源碼中發現 系統會判斷service的package是否為空,所以我們要設置package(就是遠程服務所在的報名)所以要intent.setpackage(包名)
這就完事了,大體實現了跨進程通信了
補充一下 aidl實現回調
(要之前 一直是客戶端一直通過方法去找aidl要東西,回調就實現了aidl主動給客戶端東西)
其實總體思路就是一般的接口回調(回調的思路就是在aidl設置一個接口作為參數,然后aidl中callback.callbackMethod(params)就可以了把params傳給了客戶端,接口回調不懂的可以看我的早期文章,有講的)
那么就要定義aidl接口,其實就是新建一個接口的aidl文件就行了
如圖(
)
然后就在我們的aidl設置這個接口為參數就行了,這樣就可以了
我要說注意的地方就是在registerCallback
private RemoteCallbackList<ICallback> iCallbackRemoteCallbackList;
必須要通過這個來for循環來注冊每個callback,具體請看我的Libhandler中的實現
講講為什么這樣,因為是遠程服務,所以不一定只有一個callback,所以就需要map把每個callback與每個注冊的遠程服務相綁定,真心佩服android人員設計的巧妙
總結
這就是我理解的aidl,希望對閱讀的你有所幫助,當然 這里要是有不足,歡迎指出和我聯系.