Android IPC机制Messenger实例详解

作者:左手木亽 时间:2023-04-27 13:25:06 

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,一次一个处理请求。

来源:http://blog.csdn.net/neacy_zz/article/details/50811596

标签:Android,IPC机制,Messenger
0
投稿

猜你喜欢

  • Java AtomicInteger类的使用方法详解

    2023-03-09 00:57:57
  • struts+spring+hibernate三个框架的整合

    2022-08-08 08:13:15
  • Spring框架构造注入type属性实例详解

    2021-10-04 19:45:19
  • c语言中static修饰函数的方法及代码

    2023-08-26 15:46:21
  • C#的FileInfo类实现文件操作实例

    2021-06-30 06:39:44
  • JVM完全解读之GC日志记录分析

    2022-09-22 12:43:08
  • 详解Java读取Jar中资源文件及示例代码

    2021-07-12 11:18:52
  • java httpclient设置超时时间和代理的方法

    2023-05-10 13:05:24
  • C#实现对象XML序列化的方法

    2022-02-11 01:30:59
  • Spring Boot项目@RestController使用重定向redirect方式

    2023-12-11 15:21:56
  • java自动生成ID号的方法

    2023-11-18 11:17:56
  • 通过Class类获取对象(实例讲解)

    2023-05-19 14:12:58
  • C#中使用闭包与意想不到的坑详解

    2022-08-15 00:13:58
  • Java持久化框架Hibernate与Mybatis优劣及选择详解

    2023-12-03 19:46:14
  • Java集合的总体框架相关知识总结

    2021-06-25 10:43:36
  • springboot动态定时任务的实现方法示例

    2023-04-20 15:59:58
  • Java实现角色扮演游戏的示例代码

    2023-03-31 19:41:45
  • 浅谈Java内存区域划分和内存分配策略

    2023-08-11 18:52:49
  • java中this与super关键字的使用方法

    2022-05-04 22:03:29
  • Idea里github的图形化操作配置方法

    2021-12-29 15:03:17
  • asp之家 软件编程 m.aspxhome.com