溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Android IPC機(jī)制Messenger實(shí)例詳解

發(fā)布時間:2020-08-26 13:01:05 來源:腳本之家 閱讀:121 作者:左手木亽 欄目:移動開發(fā)

Android IPC機(jī)制Messenger實(shí)例詳解

前言:

Messenger可以翻譯成信使,通過它可以在不同進(jìn)程間傳遞Message對象有了它就可以輕松實(shí)現(xiàn)進(jìn)程間的數(shù)據(jù)傳遞了。

Messenger使用的方法相對AIDL比較簡單,它對AIDL做了一層封裝是的我們不需要像采用AIDL那樣去實(shí)現(xiàn)進(jìn)程通信那么麻煩,可以看看他的源碼有AIDL的跡象。

public final class Messenger implements Parcelable {
  private final IMessenger mTarget;

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

  public void send(Message message) throws RemoteException {
    mTarget.send(message);
  }

  public IBinder getBinder() {
    return mTarget.asBinder();
  }

  public boolean equals(Object otherObj) {
    if (otherObj == null) {
      return false;
    }
    try {
      return mTarget.asBinder().equals(((Messenger)otherObj)
          .mTarget.asBinder());
    } catch (ClassCastException e) {
    }
    return false;
  }

  public int hashCode() {
    return mTarget.asBinder().hashCode();
  }

  public int describeContents() {
    return 0;
  }

  public void writeToParcel(Parcel out, int flags) {
    out.writeStrongBinder(mTarget.asBinder());
  }

  public static final Parcelable.Creator<Messenger> CREATOR
      = new Parcelable.Creator<Messenger>() {
    public Messenger createFromParcel(Parcel in) {
      IBinder target = in.readStrongBinder();
      return target != null ? new Messenger(target) : null;
    }

    public Messenger[] newArray(int size) {
      return new Messenger[size];
    }
  };

  public static void writeMessengerOrNullToParcel(Messenger messenger,
      Parcel out) {
    out.writeStrongBinder(messenger != null ? messenger.mTarget.asBinder()
        : null);
  }

  public static Messenger readMessengerOrNullFromParcel(Parcel in) {
    IBinder b = in.readStrongBinder();
    return b != null ? new Messenger(b) : null;
  }

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

首先我們需要新建一個Service來處理客戶端的請求,同時聲明一個Handler作為參數(shù)來創(chuàng)建一個Messenger,然后通過getBinder()方法返回Binder。

public class MessageService extends Service {

  private Messenger mMessenger = new Messenger(new Handler() {
    @Override
    public void handleMessage(Message msgFromClient) {
      super.handleMessage(msgFromClient);
      Message msgToTarget = Message.obtain(msgFromClient);
      msgToTarget.what = 0;
      try {
        Thread.sleep(2000);
        msgToTarget.arg1 = msgFromClient.arg1 + msgFromClient.arg2;
        msgFromClient.replyTo.send(msgToTarget);
      } catch (InterruptedException e) {
        e.printStackTrace();
      } catch (RemoteException e) {
        e.printStackTrace();
      }
    }
  });

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

里面的邏輯是簡單的將客戶端傳來的Message中的arg1和arg2的值相加并將結(jié)果返回給Message對應(yīng)的replyTo這個Messenger,并通過send將服務(wù)端的Message返回給客戶端。

然后在客戶端處理:首先需要bindService來綁定這個Service,然后通過IBinder生成一個Messenger對象,這個Messenger對象就可以將需要處理的數(shù)據(jù)封裝到Message然后send到Service去。

 Messenger mMessenger = new Messenger(new Handler() {
    @Override
    public void handleMessage(Message msg) {
      super.handleMessage(msg);
      Log.w("Jayuchou", "--- 從異步線程中讀取到數(shù)據(jù) --- " + msg.arg1);
    }
  });

  Messenger mService;

  ServiceConnection connection = new ServiceConnection() {
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
      mService = new Messenger(service);
      Log.w("Jayuchou", "-- Connected success --");
    }

    @Override
    public void onServiceDisconnected(ComponentName name) {
      Log.w("Jayuchou", "-- Connected dismiss --");
      mService = null;
    }
  };

然后調(diào)用的地方方式為:

     Message msgFromClient = Message.obtain(null, 0, 1, 2);
        msgFromClient.replyTo = mMessenger;
        try {
          mService.send(msgFromClient);
        } catch (RemoteException e) {
          e.printStackTrace();
        }

將數(shù)據(jù)封裝Message中,并且Message中的replyTo指定服務(wù)端中要將結(jié)果回調(diào)的Messenger對象。

msgFromClient.replyTo.send(msgToTarget);

我們可以看到Service中有這么一句代碼,其中的replyTo就是我們在客戶端傳進(jìn)去的Messenger,這時候調(diào)用send方法就可以將服務(wù)端的也就是另一個進(jìn)程的數(shù)據(jù)傳到想要用的進(jìn)程然后采用Messenger進(jìn)行接收,我們可以跟Handler用法類似的使用即可。Messenger是一個輕量級的AIDL,一次一個處理請求。

以上就是Android messenger 的消息處理的詳解,關(guān)于Android 開發(fā)的文章,本站還很多,請大家搜索參閱,感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI