Android中的IPC方式

跨进程通信的方式有很多,在这里一一展开。

Bundle方式

四大组件都是支持Intent中传递Bundle数据的,所以在启动另一个进程时候可以通过Intent附加Bundle数据进行传递,部分场景还可以使用IntentService,需要注意的是Bundle传递的数据要序列化。

文件共享方式

通俗的来讲就是两个进程同时来读/写一个文件来交换数据,虽然有风险,但在数据不是太频繁读和写还是可以的,具体的实现方式就是在一个进程中序列化数据到本地,然后在另一个进程中反序列化,具体实现方式在前面序列化和反序列化有提到。

Messenger方式

它是可以翻译为信使,是一种轻量级的IPC方案,可以传递Message对象。实现Messenger分为服务端和客户端:

我们先来看服务端,服务端代码如下:

package com.bigaka.service;

import android.app.Service;
import android.content.Intent;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.os.Messenger;
import android.support.annotation.Nullable;
import android.util.Log;

import com.bigaka.Utils.Constants;

/**
 * Created by ${luomingjun} on 16/9/24
 *
 */
public class MessengerService extends Service{
    public static final String TAG="MessengerService";

    private static class MessengerHandler extends Handler{
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what)
            {
                case Constants.MSG_FROM_CLIENT:
                    Log.e(TAG,"接收的消息来自于客户端:"+msg.getData().getString("msg"));
                    break;
                default:
                    super.handleMessage(msg);
            }

        }
    }

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

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

可以看到我们创建了一个Service来处理客户端的连接请求,同时创建了一个Messenger对象,并从onBind中返回了这个Messenger,别忘了在Menifest中注册这个Service。

       <service android:name="com.bigaka.service.MessengerService"
            android:process=":remote"
            />

接下来我们来看客户端:

package com.bigaka.messenger;

import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.os.Message;
import android.os.Messenger;
import android.os.RemoteException;
import android.support.v7.app.AppCompatActivity;

import com.bigaka.Utils.Constants;
import com.bigaka.service.MessengerService;

public class MainActivity extends AppCompatActivity {
    private static final String TAG="MainActivity";

    private Messenger messenger;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Intent intent=new Intent(this, MessengerService.class);
        bindService(intent,mConnection, Context.BIND_AUTO_CREATE);

    }

    private ServiceConnection mConnection=new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            messenger=new Messenger(service);
            Message msg=Message.obtain(null, Constants.MSG_FROM_CLIENT);
            Bundle bundle=new Bundle();
            bundle.putString("msg","这是从客户端发送的消息");
            msg.setData(bundle);
            try {
                messenger.send(msg);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {

        }
    };

    @Override
    protected void onDestroy() {
        super.onDestroy();
        unbindService(mConnection);
    }
}

客户端绑定了我们刚刚的MessengerService,并从绑定服务成功的IBinder对象中创建了一个Messenger,并利用Message来send消息。
运行起来之后,我们可以从日志中看到:

09-24 07:14:46.420 2944-2944/com.bigaka.messenger:remote E/MessengerService: 接收的消息来自于客户端:这是从客户端发送的消息

这种方式只是单向的,那么我现在希望我给服务端发送了一条消息,并且它需要给我回复一条消息,那又该如何改造我们现在的方法呢?

服务端:

private static class MessengerHandler extends Handler{
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what)
            {
                case Constants.MSG_FROM_CLIENT:
                    Log.e(TAG,"接收的消息来自于客户端:"+msg.getData().getString("msg"));
                    Messenger messenger=msg.replyTo;
                    Message relpyMessage=Message.obtain(null,Constants.MSG_FROM_CLIENT);
                    Bundle bundle=new Bundle();
                    bundle.putString("reply","已经收到你的消息了,这是回复你的消息");
                    relpyMessage.setData(bundle);
                    try {
                        messenger.send(relpyMessage);
                    } catch (RemoteException e) {
                        e.printStackTrace();
                    }
                    break;
                default:
                    super.handleMessage(msg);
            }

        }
    }

客户端我们需要加一个Handler来接收消息,如下:

private static class MessengerHandler extends Handler {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what)
            {
                case Constants.MSG_FROM_CLIENT:
                    Log.e(TAG, "接收的消息来自于服务端:" + msg.getData().getString("msg"));
                    break;
                default:
                    super.handleMessage(msg);
            }

        }
    }

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

可能会好奇mReplyMessenger在那个地方用呢?注意在客户端发送消息时候,需要将msg.replyTo=mReplyMessenger;看我们现在的客户端代码:

 private ServiceConnection mConnection=new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            messenger=new Messenger(service);
            Message msg=Message.obtain(null, Constants.MSG_FROM_CLIENT);
            Bundle bundle=new Bundle();
            bundle.putString("msg","这是从客户端发送的消息");
            msg.setData(bundle);

            /***  注意 注意  注意 ***/
            msg.replyTo=mReplyMessenger;
            try {
                messenger.send(msg);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {

        }

现在我们来运行一些,看下Log:

09-24 07:47:13.394 29339-29339/com.bigaka.messenger:remote E/MessengerService: 接收的消息来自于客户端:这是从客户端发送的消息
09-24 07:47:13.771 29311-29311/com.bigaka.messenger E/MainActivity: 接收的消息来自于服务端:已经收到你的消息了,这是回复你的消息

通信方式就介绍到这里了,可以扩展开来。
以上源码:https://pan.baidu.com/s/1nv6iqVJ 提取码:tfpk

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,881评论 4 368
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,052评论 1 301
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 110,598评论 0 250
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,407评论 0 217
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,823评论 3 294
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,872评论 1 224
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,037评论 2 317
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,778评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,505评论 1 247
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,745评论 2 253
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,233评论 1 264
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,568评论 3 260
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,231评论 3 241
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,141评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,939评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,954评论 2 283
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,784评论 2 275

推荐阅读更多精彩内容