Android深入理解IPC機(jī)制(二)淺談Binder

Binder是Android中的一個(gè)類蚓庭,它繼承了IBinder接口娇跟。

  1. 從IPC角度考慮僻爽,Binder是Android中的一種跨進(jìn)程通信方式,Binder還可以理解為一種虛擬的物理設(shè)備呛哟,它的設(shè)備驅(qū)動(dòng)是/dev/binder叠荠;
  2. 從Android Framework角度考慮,Binder是連接各種Manager和相應(yīng)ServiceManager的橋梁扫责;
  3. 從Android應(yīng)用層來(lái)講榛鼎,Binder是客戶端和服務(wù)器進(jìn)行通信的媒介。當(dāng)bindService的時(shí)候鳖孤,服務(wù)端會(huì)返回一個(gè)Binder對(duì)象者娱,通過(guò)這個(gè)對(duì)象,客戶端可以獲取服務(wù)端提供的數(shù)據(jù)或服務(wù)苏揣,包括普通服務(wù)和基于AIDL的服務(wù)黄鳍;

在Android開(kāi)發(fā)中,Binder主要用在Service中平匈,包括AIDL和Messenger框沟,實(shí)際上Messenger底層也是AIDL,所以選擇用AIDL來(lái)分析Binder的工作機(jī)制增炭。

首先我們新建一個(gè)AIDL忍燥,系統(tǒng)SDK會(huì)自動(dòng)為我們生成一個(gè)Binder類。
(1)新建Book.java隙姿、Book.aidl梅垄、IBookManager.aidl文件

//Book.java
package com.example.qianwei.myapplication.aidl;

import android.os.Parcel;
import android.os.Parcelable;

public class Book implements Parcelable {

    private int id;
    private String name;

    public Book(int id, String name) {
        this.id = id;
        this.name = name;
    }

    protected Book(Parcel in) {
        id = in.readInt();
        name = in.readString();
    }

    public static final Creator<Book> CREATOR = new Creator<Book>() {
        @Override
        public Book createFromParcel(Parcel in) {
            return new Book(in);
        }

        @Override
        public Book[] newArray(int size) {
            return new Book[size];
        }
    };

    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(id);
        dest.writeString(name);
    }
}
// Book.aidl
package com.example.qianwei.myapplication.aidl;

parcelable Book;
// IBookManager.aidl
package com.example.qianwei.myapplication.aidl;

//AIDL特殊之處
import com.example.qianwei.myapplication.aidl.Book;

interface IBookManager {

    /**
     * 獲取圖書列表
     */
    List<Book> getBookList();

    /**
     * 添加圖書
     */
    void addBook(in Book book);
}

正如我們所見(jiàn),盡管Book 類與IBookManager 位于同一個(gè)包下面输玷,但是我們?nèi)匀恍枰褂胕mport將Book類導(dǎo)入到IBookManager 队丝,否則編譯會(huì)出錯(cuò),這就是AIDL的特殊之處饲嗽。

我嘗試著將Book聲明注釋掉炭玫,果然編譯器在編譯階段給出了編譯失敗提示:

Process 'command 'F:\sdk\build-tools\28.0.3\aidl.exe'' finished with non-zero exit value 1

(2)編譯完成之后,我們可以看到build/generated/source/aidl目錄下的com.example.qianwei.myapplication.aidl包里面出現(xiàn)了一個(gè)IBookManager.java類貌虾,這就是我們要找的類吞加。

package com.example.qianwei.myapplication.aidl;

public interface IBookManager extends android.os.IInterface {
    /**
     * Local-side IPC implementation stub class.
     */
    public static abstract class Stub extends android.os.Binder implements
            com.example.qianwei.myapplication.aidl.IBookManager {
        private static final java.lang.String DESCRIPTOR = "com.example.qianwei.myapplication.aidl.IBookManager";

        /**
         * Construct the stub at attach it to the interface.
         */
        public Stub() {
            this.attachInterface(this, DESCRIPTOR);
        }

        /**
         * Cast an IBinder object into an com.example.qianwei.myapplication.aidl.IBookManager interface,
         * generating a proxy if needed.
         */
        public static com.example.qianwei.myapplication.aidl.IBookManager asInterface(android.os.IBinder obj) {
            if ((obj == null)) {
                return null;
            }
            android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
            if (((iin != null) && (iin instanceof com.example.qianwei.myapplication.aidl.IBookManager))) {
                return ((com.example.qianwei.myapplication.aidl.IBookManager) iin);
            }
            return new com.example.qianwei.myapplication.aidl.IBookManager.Stub.Proxy(obj);
        }

        @Override
        public android.os.IBinder asBinder() {
            return this;
        }

        @Override
        public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags)
                throws android.os.RemoteException {
            java.lang.String descriptor = DESCRIPTOR;
            switch (code) {
                case INTERFACE_TRANSACTION: {
                    reply.writeString(descriptor);
                    return true;
                }
                case TRANSACTION_getBookList: {
                    data.enforceInterface(descriptor);
                    java.util.List<com.example.qianwei.myapplication.aidl.Book> _result = this.getBookList();
                    reply.writeNoException();
                    reply.writeTypedList(_result);
                    return true;
                }
                case TRANSACTION_addBook: {
                    data.enforceInterface(descriptor);
                    com.example.qianwei.myapplication.aidl.Book _arg0;
                    if ((0 != data.readInt())) {
                        _arg0 = com.example.qianwei.myapplication.aidl.Book.CREATOR.createFromParcel(data);
                    } else {
                        _arg0 = null;
                    }
                    this.addBook(_arg0);
                    reply.writeNoException();
                    return true;
                }
                default: {
                    return super.onTransact(code, data, reply, flags);
                }
            }
        }

        private static class Proxy implements com.example.qianwei.myapplication.aidl.IBookManager {
            private android.os.IBinder mRemote;

            Proxy(android.os.IBinder remote) {
                mRemote = remote;
            }

            @Override
            public android.os.IBinder asBinder() {
                return mRemote;
            }

            public java.lang.String getInterfaceDescriptor() {
                return DESCRIPTOR;
            }

            /**
             * 獲取圖書列表
             */
            @Override
            public java.util.List<com.example.qianwei.myapplication.aidl.Book> getBookList()
                throws android.os.RemoteException {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                java.util.List<com.example.qianwei.myapplication.aidl.Book> _result;
                try {
                    _data.writeInterfaceToken(DESCRIPTOR);
                    mRemote.transact(Stub.TRANSACTION_getBookList, _data, _reply, 0);
                    _reply.readException();
                    _result = _reply.createTypedArrayList(com.example.qianwei.myapplication.aidl.Book.CREATOR);
                } finally {
                    _reply.recycle();
                    _data.recycle();
                }
                return _result;
            }

            /**
             * 添加圖書
             */
            @Override
            public void addBook(com.example.qianwei.myapplication.aidl.Book book) throws android.os.RemoteException {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                try {
                    _data.writeInterfaceToken(DESCRIPTOR);
                    if ((book != null)) {
                        _data.writeInt(1);
                        book.writeToParcel(_data, 0);
                    } else {
                        _data.writeInt(0);
                    }
                    mRemote.transact(Stub.TRANSACTION_addBook, _data, _reply, 0);
                    _reply.readException();
                } finally {
                    _reply.recycle();
                    _data.recycle();
                }
            }
        }

        static final int TRANSACTION_getBookList = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
        static final int TRANSACTION_addBook = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1);
    }

    /**
     * 獲取圖書列表
     */
    public java.util.List<com.example.qianwei.myapplication.aidl.Book> getBookList() throws
            android.os.RemoteException;

    /**
     * 添加圖書
     */
    public void addBook(com.example.qianwei.myapplication.aidl.Book book)
            throws android.os.RemoteException;
}

這個(gè)類內(nèi)容有點(diǎn)長(zhǎng),看起來(lái)似乎很混亂尽狠,我們先從整體上來(lái)逐步分析:


類結(jié)構(gòu)圖
  • 兩個(gè)抽象方法:getBookList()衔憨、addBook(com.example.qianwei.myapplication.aidl.Book book),這就是我們?cè)贗BookManager.aidl類中聲明的方法袄膏。同時(shí)還聲明了兩個(gè)整型id用于標(biāo)識(shí)這兩個(gè)方法践图,這兩個(gè)id用于判斷在transact過(guò)程中客戶端調(diào)用的是哪個(gè)方法。
  • 聲明一個(gè)內(nèi)部類Stub沉馆,也就是一個(gè)Binder類码党,當(dāng)客戶端與服務(wù)器位于同一進(jìn)程德崭,方法調(diào)用不會(huì)走跨進(jìn)程的transact過(guò)程,否則需要執(zhí)行transact過(guò)程揖盘,這個(gè)過(guò)程由Stub的內(nèi)部代理類Proxy完成眉厨。

總而言之,這個(gè)接口的核心內(nèi)容就是它的內(nèi)部類Stub和Stub的內(nèi)部代理類Proxy兽狭,我們繼續(xù)分析這兩個(gè)類:

DESCRIPTOR
Binder類的唯一標(biāo)識(shí)憾股,一般用當(dāng)前類名表示。

private static final java.lang.String DESCRIPTOR = "com.example.qianwei.myapplication.aidl.IBookManager";

asInterface(android.os.IBinder obj)
將服務(wù)端返回的Binder類型的對(duì)象轉(zhuǎn)化為客戶端所需的AIDL接口類型的對(duì)象箕慧,這個(gè)過(guò)程是區(qū)分進(jìn)程的服球,如果兩端是同一進(jìn)程,那么此方法返回的就是Stub對(duì)象本身颠焦,如果不是同一進(jìn)程斩熊,則返回系統(tǒng)封裝后的Stub.Proxy對(duì)象。

asBinder()
用于返回當(dāng)前Binder對(duì)象蒸健。

onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags)
這個(gè)方法運(yùn)行在服務(wù)端中的Binder線程池中座享,當(dāng)客戶端發(fā)起跨進(jìn)程請(qǐng)求,遠(yuǎn)程請(qǐng)求最終會(huì)交由此方法處理似忧。

  • 服務(wù)端通過(guò)code確定客戶端請(qǐng)求的目標(biāo)方法是什么
  • 服務(wù)端從data中取出目標(biāo)方法所需的參數(shù)(如果目標(biāo)方法有參數(shù))
  • 目標(biāo)方法執(zhí)行完畢后,向reply中寫入返回值(如果目標(biāo)方法有返回值)

Proxy#getBookList()
此方法運(yùn)行在客戶端盯捌,內(nèi)部實(shí)現(xiàn):首先創(chuàng)建該方法所需要的輸入型Parcel對(duì)象_data箫攀,輸出型Parcel對(duì)象_reply和返回對(duì)象List幼衰;接著調(diào)用mRemote.transact發(fā)起遠(yuǎn)程過(guò)程調(diào)用(RPC)請(qǐng)求渡嚣,同時(shí)當(dāng)前線程掛起识椰;然后服務(wù)端的onTransact方法會(huì)被調(diào)用藏畅,直到RPC過(guò)程返回后愉阎,當(dāng)前線程繼續(xù)執(zhí)行榜旦,并從_reply中取出RPC過(guò)程的返回結(jié)果_result,最后返回_result。

Proxy#addBook(com.example.qianwei.myapplication.aidl.Book book)
此方法運(yùn)行在客戶端上滨彻,執(zhí)行過(guò)程和getBookList()是一樣的亭饵,由于此方法沒(méi)有返回值,所以無(wú)需從_reply中取出取出_result踏兜。

使用Binder的時(shí)候有兩點(diǎn)需要額外注意:
1八秃、客戶端發(fā)起遠(yuǎn)程請(qǐng)求時(shí)疹尾,當(dāng)前進(jìn)程會(huì)被掛起直至服務(wù)端進(jìn)程返回?cái)?shù)據(jù)纳本,所以如果遠(yuǎn)程方法是耗時(shí)的繁成,那么就不能在UI線程發(fā)起此請(qǐng)求巾腕;
2、由于服務(wù)端的Binder方法是運(yùn)行在Binder池中的毁嗦,所以Binder方法應(yīng)該采取同步的方法去實(shí)現(xiàn)回铛,因?yàn)樗呀?jīng)運(yùn)行在一個(gè)線程中了克锣。

Binder的工作機(jī)制

接下來(lái)我們思考這樣一種場(chǎng)景焚志,當(dāng)Binder運(yùn)行在服務(wù)端進(jìn)程的時(shí)候進(jìn)程由于某些原因異常終止内列,服務(wù)端的Binder會(huì)發(fā)生連接斷裂(Binder死亡)浊猾,導(dǎo)致遠(yuǎn)程調(diào)用失敗纳寂,從而影響客戶端功能厂僧。為了解決這個(gè)問(wèn)題鸟召,Binder提供了兩個(gè)非常重要的配對(duì)方法linkToDeath和unlinkToDeath胆绊,通過(guò)linkToDeath我們可以為Binder設(shè)置死亡代理,當(dāng)Binder發(fā)生死亡的時(shí)候我們會(huì)收到通知压状,這個(gè)時(shí)候我們就可以通過(guò)一些恰當(dāng)?shù)姆绞街匦掳l(fā)起連接請(qǐng)求從而恢復(fù)連接跟继。我們要如何設(shè)置死亡代理呢?
1剩盒、首先,聲明一個(gè)接口IBinder.DeathRecipient并實(shí)現(xiàn)binderDied()方法辽聊,當(dāng)Binder死亡的時(shí)候系統(tǒng)會(huì)回調(diào)binderDied()方法异袄,這個(gè)時(shí)候我們可以移除之前綁定的binder代理并重新綁定遠(yuǎn)程服務(wù)

private IBinder.DeathRecipient deathRecipient = new IBinder.DeathRecipient() {
        @Override
        public void binderDied() {
            if (bookManager == null) {
                return;
            }
            bookManager.asBinder().unlinkToDeath(deathRecipient, 0);
            bookManager = null;
            //這里重新綁定遠(yuǎn)程Service
        }
    };

2、客戶端綁定遠(yuǎn)程服務(wù)成功后玛臂,給binder設(shè)置死亡代理

try {
    bookManager = IBookManager.Stub.asInterface(binder);
    binder.linkToDeath(deathRecipient, 0);
} catch (RemoteException e) {
    e.printStackTrace();
}

這樣我們就成功了設(shè)置了死亡代理烤蜕,另外通過(guò)Binder的方法isBinderAlive也可以判斷binder是否死亡。


推薦閱讀

Android深入理解IPC機(jī)制(一)基礎(chǔ)知識(shí)概要
Android深入理解IPC機(jī)制(三) Android中的幾種IPC方式
Android深入理解IPC機(jī)制(四)Binder連接池

參考書籍

《Android開(kāi)發(fā)藝術(shù)探索》

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末迹冤,一起剝皮案震驚了整個(gè)濱河市讽营,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌泡徙,老刑警劉巖橱鹏,帶你破解...
    沈念sama閱讀 217,734評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡莉兰,警方通過(guò)查閱死者的電腦和手機(jī)挑围,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)糖荒,“玉大人杉辙,你說(shuō)我怎么就攤上這事〈范洌” “怎么了蜘矢?”我有些...
    開(kāi)封第一講書人閱讀 164,133評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)综看。 經(jīng)常有香客問(wèn)我硼端,道長(zhǎng),這世上最難降的妖魔是什么寓搬? 我笑而不...
    開(kāi)封第一講書人閱讀 58,532評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮县耽,結(jié)果婚禮上句喷,老公的妹妹穿的比我還像新娘。我一直安慰自己兔毙,他們只是感情好唾琼,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,585評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著澎剥,像睡著了一般锡溯。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上哑姚,一...
    開(kāi)封第一講書人閱讀 51,462評(píng)論 1 302
  • 那天祭饭,我揣著相機(jī)與錄音,去河邊找鬼叙量。 笑死倡蝙,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的绞佩。 我是一名探鬼主播寺鸥,決...
    沈念sama閱讀 40,262評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼品山!你這毒婦竟也來(lái)了胆建?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 39,153評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤肘交,失蹤者是張志新(化名)和其女友劉穎笆载,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,587評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡宰译,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,792評(píng)論 3 336
  • 正文 我和宋清朗相戀三年檐蚜,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片沿侈。...
    茶點(diǎn)故事閱讀 39,919評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡闯第,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出缀拭,到底是詐尸還是另有隱情咳短,我是刑警寧澤,帶...
    沈念sama閱讀 35,635評(píng)論 5 345
  • 正文 年R本政府宣布蛛淋,位于F島的核電站咙好,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏褐荷。R本人自食惡果不足惜勾效,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,237評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望叛甫。 院中可真熱鬧层宫,春花似錦、人聲如沸其监。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,855評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)抖苦。三九已至毁菱,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間锌历,已是汗流浹背贮庞。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 32,983評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留究西,地道東北人贸伐。 一個(gè)月前我還...
    沈念sama閱讀 48,048評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像怔揩,于是被迫代替她去往敵國(guó)和親捉邢。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,864評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容