Android IPC机制Messenger实例详解

 更新时间:2017年07月15日 16:31:21   作者:左手木亽  
这篇文章主要介绍了 Android IPC机制Messenger实例详解的相关资料,需要的朋友可以参考下

Android IPC机制Messenger实例详解

前言:

Messenger可以翻译成信使,通过它可以在不同进程间传递Message对象有了它就可以轻松实现进程间的数据传递了。

Messenger使用的方法相对AIDL比较简单,它对AIDL做了一层封装是的我们不需要像采用AIDL那样去实现进程通信那么麻烦,可以看看他的源码有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作为参数来创建一个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的值相加并将结果返回给Message对应的replyTo这个Messenger,并通过send将服务端的Message返回给客户端。

然后在客户端处理:首先需要bindService来绑定这个Service,然后通过IBinder生成一个Messenger对象,这个Messenger对象就可以将需要处理的数据封装到Message然后send到Service去。

 Messenger mMessenger = new Messenger(new Handler() {
    @Override
    public void handleMessage(Message msg) {
      super.handleMessage(msg);
      Log.w("Jayuchou", "--- 从异步线程中读取到数据 --- " + 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;
    }
  };

然后调用的地方方式为:

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

将数据封装Message中,并且Message中的replyTo指定服务端中要将结果回调的Messenger对象。

msgFromClient.replyTo.send(msgToTarget);

我们可以看到Service中有这么一句代码,其中的replyTo就是我们在客户端传进去的Messenger,这时候调用send方法就可以将服务端的也就是另一个进程的数据传到想要用的进程然后采用Messenger进行接收,我们可以跟Handler用法类似的使用即可。Messenger是一个轻量级的AIDL,一次一个处理请求。

以上就是Android messenger 的消息处理的详解,关于Android 开发的文章,本站还很多,请大家搜索参阅,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

相关文章

  • Android组件化、插件化详细讲解

    Android组件化、插件化详细讲解

    这篇文章主要介绍了Android组件化、插件化详细讲解,这些单独⼆次封装的功能模块apk,就称作插件,文章围绕主题展开详细的内容介绍,需要的朋友可以参考一下
    2022-07-07
  • Android 实现云知声版离线语音合成

    Android 实现云知声版离线语音合成

    这篇文章主要介绍了Android 实现云知声版离线语音合成,目前云知声提供免费的离线TTS,功能也比较简单,合成的语音也比较生硬,如果对合成的语音要求不高的话可以考虑接入。具体合成需要的小伙伴可以参考下面文章内容
    2022-06-06
  • RecyclerView设置间距和添加分割线的方法

    RecyclerView设置间距和添加分割线的方法

    在使用RecyclerView布局,经常需要调整间距和添加分割线以达到更美观效果,这篇文章主要介绍了RecyclerView设置间距和添加分割线的方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-09-09
  • Flutter软键盘的原理浅析

    Flutter软键盘的原理浅析

    大家应该都知道目前Flutter官方是没有自定义键盘的解决方案,下面这篇文章主要给大家介绍了关于Flutter软键盘原理的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
    2021-10-10
  • android控件实现多张图片渐变切换

    android控件实现多张图片渐变切换

    这篇文章主要为大家详细介绍了android控件实现多张图片渐变切换,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-08-08
  • Android中判断listview是否滑动到顶部和底部的实现方法

    Android中判断listview是否滑动到顶部和底部的实现方法

    下面小编就为大家分享一篇Android中判断listview是否滑动到顶部和底部的实现方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2017-11-11
  • Android 仿京东商城底部布局的选择效果(Selector 选择器的实现)

    Android 仿京东商城底部布局的选择效果(Selector 选择器的实现)

    这篇文章主要介绍了Android 仿京东商城底部布局的选择效果(Selector 选择器的实现),需要的朋友可以参考下
    2017-04-04
  • 自定义AlertDialog去除黑色背景的解决方法

    自定义AlertDialog去除黑色背景的解决方法

    今天小编就为大家分享一篇自定义AlertDialog去除黑色背景的解决方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-07-07
  • Android Studio格式化(Format)代码快捷键介绍

    Android Studio格式化(Format)代码快捷键介绍

    这篇文章主要介绍了Android Studio格式化(Format)代码快捷键,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-01-01
  • Android编程中读写私有文件的方法

    Android编程中读写私有文件的方法

    这篇文章主要介绍了Android编程中读写私有文件的方法,结合实例形式分析了Android针对私有文件读写操作相关技巧,需要的朋友可以参考下
    2016-10-10

最新评论