ipc 使用Messenger進(jìn)行跨進(jìn)程通信

Messenger

Messenger翻譯為信使,即它可以在不同進(jìn)程中傳遞Message對象。在Message中放入我們需要傳遞的數(shù)據(jù),就可以輕松地實(shí)現(xiàn)地實(shí)現(xiàn)數(shù)據(jù)的數(shù)據(jù)間傳遞了。它的底層實(shí)現(xiàn)是AIDL。從構(gòu)造函數(shù)可以明顯看到AILD的痕跡,不管是IMessenger還是Stub.asInterface這種使用方法都是表面它的底層的AIDL。

    public Messenger(Handler target) {
        mTarget = target.getIMessenger();
    }

    public Messenger(IBinder target) {
        mTarget = IMessenger.Stub.asInterface(target);
    }

使用Messenger進(jìn)行跨進(jìn)程通信

  • 服務(wù)端進(jìn)程
    需要在服務(wù)端創(chuàng)建一個(gè)Service來處理客戶端的連接請求,同時(shí)創(chuàng)建一個(gè)Handler并通過它來創(chuàng)建一個(gè)Messenger對象,然后在Service的onBind中返回這個(gè)Messenger對象底層的Binder即可。
  • 客戶端進(jìn)程中,首先要綁定服務(wù)端的Service,綁定成功后用服務(wù)端返回的IBinder對象創(chuàng)建一個(gè)Messenger,通過這個(gè)Messenger就可以向服務(wù)端發(fā)送消息了。發(fā)送消息類型是Messeage對象。如果需要服務(wù)端能夠回應(yīng)客戶端,就和服務(wù)端一樣,我們還需要?jiǎng)?chuàng)建一個(gè)Handler并創(chuàng)建一個(gè)新的Messenger,并把這個(gè)Messenger對象通過Message的replyTo參數(shù)傳遞給服務(wù)器,服務(wù)器通過這個(gè)replyTo參數(shù)就可以回應(yīng)客戶端。

服務(wù)端的代碼

public class MessengerService extends Service {
    private static final String TAG = "MessengerService";
    private static class MessengerHandler extends Handler {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
            case MyConstants.MSG_FROM_CLIENT:
                Log.i(TAG, "來自客戶端的消息" + msg.getData().getString("msg"));
                Messenger client = msg.replyTo;
                Message relpyMessage = Message.obtain(null, MyConstants.MSG_FROM_SERVICE);
                Bundle bundle = new Bundle();
                bundle.putString("reply", "嗯,你的消息我已經(jīng)收到,稍后會回復(fù)你。");
                relpyMessage.setData(bundle);
                try {
                    client.send(relpyMessage);
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
                break;
            default:
                super.handleMessage(msg);
            }
        }
    }

    private final Messenger mMessenger = new Messenger(new MessengerHandler());

    @Override
    public IBinder onBind(Intent intent) {
        return mMessenger.getBinder();
    }

    @Override
    public void onCreate() {
        super.onCreate();
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        return super.onStartCommand(intent, flags, startId);
    }
    
}


解析:可以看到MessengerHandler 是用來處理客戶端發(fā)送過來的消息,mMessenger 是一個(gè)Messenger對象,它和MessengerHandler相關(guān)聯(lián),并在onBind方法中返回它里面的Binder對象,可以看出,這里的Messenger的作用是將客戶端發(fā)送的消息傳遞給MessengerHandler處理。注意,注冊service,讓其運(yùn)行在單獨(dú)的進(jìn)程中:

        <service
            android:name=".Messenger.MessengerService"
            android:process=":remote"
            android:exported="true">
            <intent-filter>
                <action android:name="com.lu.ipc.remote" />
            </intent-filter>
        </service>


客戶端的代碼

在onCreate()中綁定遠(yuǎn)程進(jìn)程的MessengerService。

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Intent intent = new Intent("com.lu.ipc.remote");
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP)
         intent=createExplicitFromImplicitIntent(this,intent);
        bindService(intent, mConnection, Context.BIND_AUTO_CREATE);

    }

綁定成功后根據(jù)服務(wù)端返回的binder對象創(chuàng)建Messenger對象并使用此對象向服務(wù)端發(fā)送消息,代碼中創(chuàng)建的Bundle包括了文本消息和客戶端的Messenger對象,此對象可以是接收來自服務(wù)端的消息。

    private ServiceConnection mConnection = new ServiceConnection() {
        public void onServiceConnected(ComponentName className, IBinder service) {
            mService = new Messenger(service);
            Log.d(TAG, "bind service");
            Message msg = Message.obtain(null, MyConstants.MSG_FROM_CLIENT);
            Bundle data = new Bundle();
            data.putString("msg", "從服務(wù)端發(fā)送消息");
            msg.setData(data);
            msg.replyTo = mGetReplyMessenger;
            try {
                mService.send(msg);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }

        public void onServiceDisconnected(ComponentName className) {
        }
    };

接收來自服務(wù)端的消息


    private Messenger mGetReplyMessenger = new Messenger(new MessengerHandler());
    
    private static class MessengerHandler extends Handler {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
            case MyConstants.MSG_FROM_SERVICE:
                Log.i(TAG, "接收到來自服務(wù)端的消息:" + msg.getData().getString("reply"));
                break;
            default:
                super.handleMessage(msg);
            }
        }
    }



總結(jié)

上面的例子是個(gè)很經(jīng)典的使用Messenger進(jìn)行跨進(jìn)程通信的例子,所以是在同一個(gè)應(yīng)用中,但是服務(wù)端是在不同的進(jìn)程的,這個(gè)和分別在兩個(gè)不同的應(yīng)用道理是一樣的,有興趣可以自己試試。在Messenger中進(jìn)行數(shù)據(jù)傳遞必須將數(shù)據(jù)放入Message中,而Messenger和Message都實(shí)現(xiàn)了Parcelable接口,因此可以進(jìn)行跨進(jìn)程的通信。

項(xiàng)目地址

MessengerDemo

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容