Android 的進程間通信 Binder——AIDL的入門使用(二)

進程間通信系列

AIDL的入門使用(一)
AIDL的入門使用(二)
AIDL的入門使用(三)
Messenger的入門使用

序言:

Android 的進程間通信 Binder——AIDL的入門使用(一)中我們可以通過AIDL調(diào)用服務端的方法進行操作矛洞,那可不可以反過來呢挂谍,服務端調(diào)用客戶端的方法夫植,場景:圖書館有新書時自動通知所有訂閱的讀者;這里就可以使用觀察者模式尿赚,客戶端在服務端注冊一個接口,當服務端有新書,自動調(diào)用客戶端注冊的接口。這種方式也可以用于消息推送的通知其他進程(猜測)祝谚。

對服務端進行的改造:(過程參考《Android開發(fā)藝術(shù)探索》)

1、定義一個當有新書到來時的通知接口酣衷,由于需在客戶端回調(diào)使用到了跨進程交惯,所以需要定義在AIDL文件中。

// IOnNewBookArrivedListener.aidl
package com.ljp.aidl_server.aidl;
// Declare any non-default types here with import statements
import com.ljp.aidl_server.aidl.Book;//雖然在同一個包中穿仪,但還是要進行導包操作席爽,否則會報錯。
interface IOnNewBookArrivedListener {//當有新書的時候的通知接口啊片。
    void onNewBookArrived(in Book newBook);
}

2只锻、在IMyAidlInterface.aidl文件中增加注冊和解除注冊的方法。

// IMyAidlInterface.aidl
package com.ljp.aidl_server.aidl;
// Declare any non-default types here with import statements
import com.ljp.aidl_server.aidl.Book;//雖然在同一個包中紫谷,但還是要進行導包操作齐饮,否則會報錯。
import com.ljp.aidl_server.aidl.IOnNewBookArrivedListener;
interface IMyAidlInterface {
    ........
    void registerListener(in IOnNewBookArrivedListener listener);// 注冊監(jiān)聽
    void unregisterListener(in IOnNewBookArrivedListener listener);//解除注冊
}

3笤昨、在AidlSerVerService中的IMyAidlInterface.Stub實現(xiàn)registerListener和unregisterListener方法祖驱,在onCreat方法中開啟一個線程,每隔5秒就加入一本新書并通知已注冊的客戶端瞒窒。

package com.ljp.aidl_server.aidl;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteCallbackList;
import android.os.RemoteException;
import android.text.TextUtils;
import android.util.Log;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
public class AidlSerVerService extends Service {
    private static final String TAG = "AidlSerVerService";
    private volatile boolean mIsServiceDestoryed = false;//Service是否銷毀
    // 用于保存跨進程接口回調(diào)的對象捺僻,已自動實現(xiàn)了線程同步
    private RemoteCallbackList<IOnNewBookArrivedListener> mListenerList = new RemoteCallbackList<>();
    //CopyOnWriteArrayList支持并發(fā)讀寫并自動進行線程同步 ,
    private CopyOnWriteArrayList<Book> mBookList = new CopyOnWriteArrayList<>();
    private String tag = "empty";
    private int num = -1;
    private IMyAidlInterface.Stub stub_binder = new IMyAidlInterface.Stub() {//IMyAidlInterface.Stub為Binder的子類
        @Override
        public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, String aString) throws RemoteException {
        }
        @Override
        public void addBook(Book book) throws RemoteException {
            addBookNotify(book);
        }
        @Override
        public List<Book> getBooks() throws RemoteException {
            return mBookList;//這里雖然返回的是CopyOnWriteArrayList,但底層會按照ArrayList進行讀取然后返回給客戶端
        }
        @Override
        public void registerListener(com.ljp.aidl_server.aidl.IOnNewBookArrivedListener listener) throws RemoteException {
            mListenerList.register(listener);
            Log.e(TAG, "registerListener: 已經(jīng)注冊Size=" + mListenerList.getRegisteredCallbackCount());
        }
        @Override
        public void unregisterListener(com.ljp.aidl_server.aidl.IOnNewBookArrivedListener listener) throws RemoteException {
            mListenerList.unregister(listener);
            Log.e(TAG, "unregisterListener: 現(xiàn)注冊的Size=" + mListenerList.getRegisteredCallbackCount());
        }
        @Override
        public void setTag(String tag) throws RemoteException {
            synchronized (this) {//進行同步操作匕坯,有可能并發(fā)
                if (!TextUtils.isEmpty(tag)) {
                    AidlSerVerService.this.tag = tag;
                }
            }
        }
        @Override
        public String getTag() throws RemoteException {
            return tag;
        }
        @Override
        public void setNum(int num) throws RemoteException {
            synchronized (this) {
                AidlSerVerService.this.num = num;
            }
        }
        @Override
        public int getNum() throws RemoteException {
            return num;
        }
    };
    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        return stub_binder;
    }
    @Override
    public void onCreate() {
        super.onCreate();
        mBookList.add(new Book(100, "Android開發(fā)", 20));
        mBookList.add(new Book(101, "Java開發(fā)", 22));
        new Thread(new Worker()).start(); //開啟一個線程束昵,每5秒添加一本新書并通知到已注冊的全部客戶端
        Log.e("aidl_server_Service", "onCreate: ");
    }
    @Override
    public void onDestroy() {
        mIsServiceDestoryed=true;
        super.onDestroy();
    }
    /**
     * 添加書籍并通知客戶端
     * @param newBook   添加的新書
     * @throws RemoteException
     */
    private void addBookNotify(Book newBook) throws RemoteException {
        mBookList.add(newBook);
        int count = mListenerList.beginBroadcast();//必須與 finishBroadcast()方法配對使用
        Log.e(TAG, "addBookNotify: 通知所有的監(jiān)聽器,size= " + count);
        for (int i = 0; i < count; i++) {
            IOnNewBookArrivedListener item = mListenerList.getBroadcastItem(i);
            if (item != null) {
                item.onNewBookArrived(newBook);//調(diào)用客戶端的方法醒颖,通知客戶端妻怎,由于調(diào)用了客戶端的方法為耗時操作建議放在子線程中
            }
        }
        mListenerList.finishBroadcast();//必須與 beginBroadcast()方法配對使用
    }
    private class Worker implements Runnable {
        @Override
        public void run() {
            while (!mIsServiceDestoryed) {
                try {
                    Thread.sleep(5 * 1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                int bookId = mBookList.size() + 1;
                Book newBook = new Book(bookId, "new book #" + bookId, bookId * 1.5);
                try {
                    addBookNotify(newBook);
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
上面使用到了CopyOnWriteArrayList和RemoteCallbackList,這里的監(jiān)聽器管理不能使用List泞歉,否則在最后解除注冊不會成功(因為服務端收到的listener每次都是新創(chuàng)建的對象)逼侦,因此采用RemoteCallbackList管理。
CopyOnWriteArrayList:支持并發(fā)讀寫腰耙,自動進行線程同步榛丢,使用和ArrayList相同,實現(xiàn)了List接口挺庞,但和ArrayList沒有任何關(guān)系晰赞。類似的還有ConcurrentHashMap
RemoteCallbackList:系統(tǒng)專門提供用于刪除跨進程listener的接口,使用了泛型支持管理任意的AIDL接口选侨,其內(nèi)部采用ArrayMap<IBinder, Callback>鍵值對的方式存儲掖鱼,當客戶端進程終止后它能自動移除客戶端已注冊的listener,援制,內(nèi)部已自動實現(xiàn)了線程同步戏挡,beginBroadcast()必須與 finishBroadcast()方法配對使用

對客戶端的改造:

將服務端的AIDL包復制到客戶端的對應位置下,創(chuàng)建一個監(jiān)聽器對象在綁定服務端成功后注冊晨仑,在onDestory方法中解除注冊褐墅。

    @Override
    protected void onDestroy() {
        UnbindAidlService(null);
        super.onDestroy();
    }
    private IMyAidlInterface mService_face;
    private static final String TAG = "Main_Client";
    ServiceConnection mConnection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            Log.e(TAG, "onServiceConnected: ");
            mService_face = IMyAidlInterface.Stub.asInterface(service);
            try {
                service.linkToDeath(mDeathRecipient, 0);//客戶端遠程綁定服務成功后,給binder設(shè)置死亡代理洪己,當服務端的binder對象死亡時系統(tǒng)回調(diào)mDeathRecipient.binderDied()方法妥凳,service.isBinderAlive();//可以判斷服務端的Binder是否死亡
                mService_face.registerListener(mIOnNewBookArrivedListener);//2、綁定成功后注冊通知的監(jiān)聽器答捕。
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.e(TAG, "onServiceDisconnected: ");
        }
    };
    private IBinder.DeathRecipient mDeathRecipient = new IBinder.DeathRecipient() {
        @Override
        public void binderDied() {
            //當Binder死亡時逝钥,系統(tǒng)會回調(diào)該方法,在此移除之前綁定的Binder代理并重新綁定遠程服務
            if (mService_face == null) return;
            mService_face.asBinder().unlinkToDeath(mDeathRecipient, 0);
            mService_face = null;
            Log.e(TAG, "binderDied: Binder死亡時,移除之前綁定的Binder代理并重新綁定遠程服務");
            bindAidlService(null);
        }
    };
    //1拱镐、創(chuàng)建一個監(jiān)聽器用于服務端添加新書時的通知晌缘。
    private IOnNewBookArrivedListener mIOnNewBookArrivedListener=new IOnNewBookArrivedListener.Stub(){
        @Override
        public void onNewBookArrived(Book newBook) throws RemoteException {
            Log.e(TAG, "onNewBookArrived: 收到了服務端更新的通知,newbook="+newBook );
        }
    };
    public void bindAidlService(View view) {
        Intent intent_service = new Intent();
        intent_service.setPackage("com.ljp.aidl_server"); //設(shè)置需要綁定的服務端的包名痢站,不是服務端Service的包名
        intent_service.setAction("server.aidl.service.action");//設(shè)置你所需調(diào)用服務的意圖
        boolean successful = bindService(intent_service, mConnection, BIND_AUTO_CREATE);
        Log.e(TAG, "bindAidlService: successful=" + successful);
    }
    public void UnbindAidlService(View view) {
        //3、客戶端銷毀時解除注冊在服務端的監(jiān)聽器
        if (mService_face!=null&&mService_face.asBinder().isBinderAlive()){
            try {
                mService_face.unregisterListener(mIOnNewBookArrivedListener);//解除在服務端的注冊
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
        if (mConnection != null) {
            unbindService(mConnection);
            Log.e(TAG, "UnbindAidlService: ");
        }
    }

測試結(jié)果:

12-01 10:59:12.689 4143-4143/com.ljp.aidl_client E/aidlLog: bindAidlService: successful=true
12-01 10:59:12.719 4143-4143/com.ljp.aidl_client E/aidlLog: onServiceConnected: 
12-01 10:59:12.719 4352-4364/com.ljp.aidl_server:remote E/aidlLog: registerListener: 已經(jīng)注冊Size=1
12-01 10:59:16.249 4352-4363/com.ljp.aidl_server:remote E/aidlLog: addBookNotify: 通知所有的監(jiān)聽器选酗,size= 1
12-01 10:59:16.249 4143-4143/com.ljp.aidl_client E/aidlLog: onNewBookArrived: 收到了服務端更新的通知阵难,newbook=Book{id=0, name='book0', price=30.5}
12-01 10:59:16.249 4143-4143/com.ljp.aidl_client E/aidlLog: addBook_AidlService: 
12-01 10:59:17.709 4352-5093/com.ljp.aidl_server:remote E/aidlLog: addBookNotify: 通知所有的監(jiān)聽器,size= 1
12-01 10:59:17.709 4143-4158/com.ljp.aidl_client E/aidlLog: onNewBookArrived: 收到了服務端更新的通知芒填,newbook=Book{id=4, name='new book #4', price=6.0}
12-01 10:59:22.699 4352-5093/com.ljp.aidl_server:remote E/aidlLog: addBookNotify: 通知所有的監(jiān)聽器呜叫,size= 1
12-01 10:59:22.709 4143-4157/com.ljp.aidl_client E/aidlLog: onNewBookArrived: 收到了服務端更新的通知空繁,newbook=Book{id=5, name='new book #5', price=7.5}
12-01 10:59:24.879 4352-4364/com.ljp.aidl_server:remote E/aidlLog: unregisterListener: 現(xiàn)注冊的Size=0
12-01 10:59:24.879 4143-4143/com.ljp.aidl_client E/aidlLog: UnbindAidlService: 
12-01 10:59:27.709 4352-5093/com.ljp.aidl_server:remote E/aidlLog: addBookNotify: 通知所有的監(jiān)聽器,size= 0

我的CSDN博客地址:http://blog.csdn.net/wo_ha/article/details/78684695

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末朱庆,一起剝皮案震驚了整個濱河市盛泡,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌娱颊,老刑警劉巖傲诵,帶你破解...
    沈念sama閱讀 217,542評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異箱硕,居然都是意外死亡拴竹,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,822評論 3 394
  • 文/潘曉璐 我一進店門剧罩,熙熙樓的掌柜王于貴愁眉苦臉地迎上來栓拜,“玉大人,你說我怎么就攤上這事惠昔∧挥耄” “怎么了?”我有些...
    開封第一講書人閱讀 163,912評論 0 354
  • 文/不壞的土叔 我叫張陵镇防,是天一觀的道長啦鸣。 經(jīng)常有香客問我,道長营罢,這世上最難降的妖魔是什么赏陵? 我笑而不...
    開封第一講書人閱讀 58,449評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮饲漾,結(jié)果婚禮上蝙搔,老公的妹妹穿的比我還像新娘。我一直安慰自己考传,他們只是感情好吃型,可當我...
    茶點故事閱讀 67,500評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著僚楞,像睡著了一般勤晚。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上泉褐,一...
    開封第一講書人閱讀 51,370評論 1 302
  • 那天赐写,我揣著相機與錄音,去河邊找鬼膜赃。 笑死挺邀,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播端铛,決...
    沈念sama閱讀 40,193評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼泣矛,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了禾蚕?” 一聲冷哼從身側(cè)響起您朽,我...
    開封第一講書人閱讀 39,074評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎换淆,沒想到半個月后哗总,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,505評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡产舞,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,722評論 3 335
  • 正文 我和宋清朗相戀三年魂奥,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片易猫。...
    茶點故事閱讀 39,841評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡耻煤,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出准颓,到底是詐尸還是另有隱情哈蝇,我是刑警寧澤,帶...
    沈念sama閱讀 35,569評論 5 345
  • 正文 年R本政府宣布攘已,位于F島的核電站炮赦,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏样勃。R本人自食惡果不足惜吠勘,卻給世界環(huán)境...
    茶點故事閱讀 41,168評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望峡眶。 院中可真熱鬧剧防,春花似錦、人聲如沸辫樱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,783評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽狮暑。三九已至鸡挠,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間搬男,已是汗流浹背拣展。 一陣腳步聲響...
    開封第一講書人閱讀 32,918評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留缔逛,地道東北人瞎惫。 一個月前我還...
    沈念sama閱讀 47,962評論 2 370
  • 正文 我出身青樓溜腐,卻偏偏與公主長得像,于是被迫代替她去往敵國和親瓜喇。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,781評論 2 354