Android IPC机制

Android IPC机制

Android中多进程模式

如何开启多进程模式

在Android中只有一种方法,那就是在AndroidMenifest文件中给四大组件(Activity,Service,Receiver,ContentProvider)指定android:process属性

  • 使用:的进程是当前应用的私有进程,其他应用组件不能和它跑在同一个进程
  • 使用.是全局进程,其他应用通过ShareUID方式可以和它跑在同一个进程
<application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity
            android:name=".MessengerActivity"
            android:launchMode="standard">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <!--使用点,要指定全包名 -->
        <activity
            android:name=".SecondActivity"

            android:process=":rcom.zhc.androidarttravel.remote">
        </activity>
        <!--使用冒号,默认会补全包名,与上述activity不是同一个进程 -->
        <service
            android:name=".service.MessengerService"
            android:process=":remote" />
        <activity android:name=".MainActivity"></activity>
    </application>

多进程模式的运行机制

  • Android为每一个进程分配了一个独立的虚拟机,具有不同的地址空间,导致每个虚拟机访问同一个类会产生多个副本,也就是说两个虚拟机数据、对象不共享,各用个的(在一个进程中改变了值,在另一个进程中不会改变,是两个单独的空间)

问题:

  1. 静态成员和单例设计模式完全失效
  2. 线程同步机制完全失效(不同进程锁的对象不同)
  3. SharedPreferences的可靠性降低(不支持两个进程同时执行写操作,会导致数据丢失)
  4. Application会多次创建(不同进程是不同的虚拟机,就是需要重新启动应用,所以创建新的Application)

IPC基础介绍

Serializable与Parcelable

区别:

  • Serializable是Java中的序列化接口,使用简单但是开销大,因为序列化和反序列化是需要大量的I/O操作
  • Parcelable是Android中的序列化方式,高效但稍显麻烦,首选Parcelable
  • 但是在序列化存储到设备中或将对象序列化后通过网络传输还是用Serializable,更方便

Serializable接口

只需要在类中声明一个serialVersionUID,也可以不需要指定UID,这个UID是在反序列化的时候判断一下是否是相同的UID对象,相同才可以反序列化

  • 这种方式要避免并发读写,内容可能不是最新,

  • 该方式适合对数据同步要求不高的进程之间通信

    //值随便取,唯一即可
    private static final long serialVersionUID = 2313186133168643352l;

序列化过程,采用ObjectOutputStreamObjectInputStream,例如下:

//序列化
        try {
            User user = new User("zhc",007,true);
        
            ObjectOutputStream out = new ObjectOutputStream(
                        new FileOutputStream("cache.txt"));
            out.writeObject(user);
            out.close();
        } catch (IOException e) {
            e.printStackTrace();
        }

//反序列化

        try {
            ObjectInputStream in = new ObjectInputStream(
                    new FileInputStream("cache.txt"));
            User newUser = (User) in.readObject();
            in.close();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }

反序列化后的对象内容和之前的一样,但是是两个不同的对象

Parcelable

先定义好变量和构造方法再实现接口

public class UserP implements Parcelable {

    private String userName;
    private int userID;
    private boolean isMale;

    public UserP(String userName, int userID, boolean isMale) {
        this.userName = userName;
        this.userID = userID;
        this.isMale = isMale;
    }

    protected UserP(Parcel in) {
        userName = in.readString();
        userID = in.readInt();
        isMale = in.readByte() != 0;
    }
    
    //反序列化,配合以上方法
    public static final Creator<UserP> CREATOR = new Creator<UserP>() {
        @Override
        public UserP createFromParcel(Parcel in) {
            return new UserP(in);
        }

        @Override
        public UserP[] newArray(int size) {
            return new UserP[size];
        }
    };

    @Override
    public int describeContents() {
        return 0;
    }

    //序列化
    @Override
    public void writeToParcel(Parcel parcel, int i) {
        parcel.writeString(userName);
        parcel.writeInt(userID);
        parcel.writeByte((byte) (isMale ? 1 : 0));
    }

}

Android中的IPC方式

使用Bundle

四大组件中的三大组件都可以用Bundle来传递数据,对象序列化之后也可以放入Bundle来传递,基本数据类型和String都是实现了序列化的。

使用文件共享

由于Android是基于Linux的,使得其并发读/写文件可以没有限制的进行,写序列化文件到SD卡中存储,另一个进程读取

    //序列化
    public void persistToFile() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                User user = new User("zhc",007,true);

                File dir = new File(MyConstants.CHAPTER_2_PATH);
                if (dir != null){
                    dir.mkdirs();
                }
                ObjectOutputStream outputStream = null;
                File cacheFile = new File(MyConstants.CACHE_FILE_PATH);
                try {
                    outputStream =
                            new ObjectOutputStream(new FileOutputStream(cacheFile));
                    outputStream.writeObject(user);

                    Log.d("user","persist File : " + user);

                } catch (IOException e) {
                    e.printStackTrace();
                }finally {
                    MyUtils.close(outputStream);
                }
            }
        }).start();
    }

    //反序列化
     public void recoverFromFile() {
    
            new Thread(new Runnable() {
                @Override
                public void run() {
                    User newUser = null;
    
                    ObjectInputStream objectInputStream = null;
    
                    File cacheFile = new File(MyConstants.CACHE_FILE_PATH);
    
                    if (cacheFile.exists()) {
                        try {
                            objectInputStream = new ObjectInputStream(
                                    new FileInputStream(cacheFile));
    
                            newUser = (User) objectInputStream.readObject();
    
                            Log.d("user", "recover from File : " + newUser);
    
                        } catch (IOException e) {
                            e.printStackTrace();
                        } catch (ClassNotFoundException e) {
                            e.printStackTrace();
                        } finally {
                            MyUtils.close(objectInputStream);
                        }
                    }
                }
            }).start();
    
        }
    

使用Messenger

Messenger是信使,他可以在不同进程中传递message对象,底层实现是AIDL,对AIDL进行了封装。它一次只处理一个请求,因此在服务端不用考虑线程同步的问题。

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

推荐阅读更多精彩内容