Android之使用文件进行IPC

简介:


一、文件进行IPC介绍

共享文件也是一种不错的进程间通信方式,两个进程通过读/写同一个文件来交换数据。在Windows上,一个文件如果被加了排斥锁将会导致其他线程无法对其进行访问,包括读写,而由于Android系统基于Linux,使其并发读/写文件可以没有限制地进行,甚至两个线程同时对同一个文件进行读写操作是允许的,尽管这可能出现问题。通过文件交换数据很好使用,除了可以交换一些文本信息外,还可以序列化一个对象到文件系统中的同时从另一个进程中恢复这个对象。

二、使用方法

1.数据类实现Parcelable或Serializable接口

public class User implements Parcelable, Serializable {public User() {
    }    public User(int userId, String userName, boolean isMale) {
        ...
    }

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

    @Override    public void writeToParcel(Parcel dest, int flags) {
        ...
    }    public static final Parcelable.Creator<User> CREATOR = new Parcelable.Creator<User>() {
        @Override        public User createFromParcel(Parcel source) {            return ...;
        }

        @Override        public User[] newArray(int size) {            return ...;
        }
    };    private User(Parcel in) {
        ...
    }

    @Override    public String toString() {        return ...;
    }
}

2.序列化一个对象到sd卡上的一个文件里

private void persistToFile() {        new Thread(new Runnable() {
            @Override            public void run() {
                User user = new User(1, "hello world", false);
                File dir = new File(MyConstants.CHAPTER_2_PATH);                if (!dir.exists()) {
                    dir.mkdirs();
                }
                File cachedFile = new File(MyConstants.CACHE_FILE_PATH);
                ObjectOutputStream objectOutputStream = null;                try {
                    objectOutputStream = new ObjectOutputStream(new FileOutputStream(cachedFile));
                    objectOutputStream.writeObject(user);
                    Log.d(TAG, "persist user:" + user);
                    mTextView.setText("persist user:" + user);
                } catch (IOException e) {
                    e.printStackTrace();
                } finally {
                    MyUtils.close(objectOutputStream);
                }
            }
        }).start();
    }

3.在另外的进程中反序列化

private void recoverFromFile(){        new Thread(new Runnable() {
            @Override            public void run() {
                User user = null;
                File cachedFile = new File(MyConstants.CACHE_FILE_PATH);                if(cachedFile.exists()){
                    ObjectInputStream objectInputStream = null;                    try{
                        objectInputStream = new ObjectInputStream(new FileInputStream(cachedFile));
                        user = (User)objectInputStream.readObject();
                        Log.d(TAG,"recover user:"+user);
                        mTextView.setText("recover user:"+user);
                    }catch (IOException e) {
                        e.printStackTrace();
                    }catch (ClassNotFoundException e){
                        e.printStackTrace();
                    }finally {
                        MyUtils.close(objectInputStream);
                    }
                }
            }
        }).start();
    }

4.在AndroidManifest.xml中开启多进程

       <activity
            ...
            android:process=":remote" />

三、小案例

1.修改activity_main.xml文件

<?xml version="1.0" encoding="utf-8"?><LinearLayout    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true"
    tools:context="com.zhangmiao.ipcdemo.MainActivity"
    android:orientation="vertical"
    >
    <TextView        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"
        android:text="File">
    </TextView>
    <Button        android:id="@+id/button1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="open activity B"
        />

    <TextView        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="A">
    </TextView></LinearLayout>

2.添加activity_second.xml文件

<?xml version="1.0" encoding="utf-8"?><LinearLayout    xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="at activity B"
        android:layout_gravity="center_horizontal"
        />
    <TextView        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Activity B"
        /></LinearLayout>

3.添加MyUtils类与MyConstants类(辅助类)

package com.zhangmiao.ipcdemo;import android.app.ActivityManager;import android.content.Context;import java.io.Closeable;import java.io.IOException;import java.util.List;import java.util.concurrent.RecursiveTask;/**
 * Created by zhangmiao on 2016/12/26. */public class MyUtils {    public static void close(Closeable closeable) {        try {            if (closeable != null) {
                closeable.close();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

package com.zhangmiao.ipcdemo;import android.os.Environment;/**
 * Created by zhangmiao on 2016/12/26. */public class MyConstants {    public static final String CHAPTER_2_PATH = Environment.getExternalStorageDirectory().getPath() + "/zhangmiao/charpter_2/";    public static final String CACHE_FILE_PATH = CHAPTER_2_PATH + "usercache";    public static final int MSG_FROM_CLIENT = 0;    public static final int MSG_FROM_SERVICE = 1;
}

4.添加User类

package com.zhangmiao.ipcdemo;import android.os.Parcel;import android.os.Parcelable;import java.io.Serializable;/**
 * Created by zhangmiao on 2016/12/26. */public class User implements Parcelable, Serializable {    public static int sUserId = 1;    private int userId;    private String userName;    private Boolean isMale;    public User() {
    }    public User(int userId, String userName, boolean isMale) {        this.userId = userId;        this.userName = userName;        this.isMale = isMale;
    }

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

    @Override    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(userId);
        dest.writeString(userName);
        dest.writeInt(isMale ? 1 : 0);
    }    public static final Parcelable.Creator<User> CREATOR = new Parcelable.Creator<User>() {
        @Override        public User createFromParcel(Parcel source) {            return new User(source);
        }

        @Override        public User[] newArray(int size) {            return new User[size];
        }
    };    private User(Parcel in) {
        userId = in.readInt();
        userName = in.readString();
        isMale = in.readInt() == 1;
    }

    @Override    public String toString() {        return String.format(                "User:{userId:%s, userName:%s, isMale:%s},",userId, userName, isMale
        );
    }
}

5.添加SecondActivity类

package com.zhangmiao.ipcdemo;import android.app.Activity;import android.os.Bundle;import android.util.Log;import android.widget.TextView;import java.io.File;import java.io.FileInputStream;import java.io.IOException;import java.io.ObjectInputStream;/**
 * Created by zhangmiao on 2016/12/26. */public class SecondActivity extends Activity {    private static final String TAG = "SecondActivity";    private TextView mTextView;

    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_second);
        Log.d(TAG, "onCreate");
        mTextView = (TextView)findViewById(R.id.textView1);
    }

    @Override    protected void onResume() {
        Log.d(TAG,"onResume");        super.onResume();
        User user = (User)getIntent().getSerializableExtra("extra_user");
        Log.d(TAG,"user:"+user.toString());
        recoverFromFile();
    }    private void recoverFromFile(){        new Thread(new Runnable() {
            @Override            public void run() {
                User user = null;
                File cachedFile = new File(MyConstants.CACHE_FILE_PATH);                if(cachedFile.exists()){
                    ObjectInputStream objectInputStream = null;                    try{
                        objectInputStream = new ObjectInputStream(new FileInputStream(cachedFile));
                        user = (User)objectInputStream.readObject();
                        Log.d(TAG,"recover user:"+user);
                        mTextView.setText("recover user:"+user);
                    }catch (IOException e) {
                        e.printStackTrace();
                    }catch (ClassNotFoundException e){
                        e.printStackTrace();
                    }finally {
                        MyUtils.close(objectInputStream);
                    }
                }
            }
        }).start();
    }
}

6.修改MainActivity类

package com.zhangmiao.ipcdemo;import android.content.ComponentName;import android.content.Context;import android.content.Intent;import android.content.ServiceConnection;import android.os.Bundle;import android.os.Handler;import android.os.IBinder;import android.os.Message;import android.os.Messenger;import android.os.RemoteException;import android.support.v7.app.AppCompatActivity;import android.util.Log;import android.view.View;import android.widget.Button;import android.widget.TextView;import java.io.File;import java.io.FileOutputStream;import java.io.IOException;import java.io.ObjectOutputStream;import java.io.Serializable;public class MainActivity extends AppCompatActivity {    private static final String TAG = "MainActivity";    private TextView mTextView;

    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        User.sUserId = 2;
        findViewById(R.id.button1).setOnClickListener(new View.OnClickListener() {
            @Override            public void onClick(View v) {
                Intent intent = new Intent();
                intent.setClass(MainActivity.this, SecondActivity.class);
                User user = new User(0, "jake", true);
                intent.putExtra("extra_user", (Serializable) user);
                startActivity(intent);
            }
        });
        mTextView = (TextView) findViewById(R.id.textView1);
    }

    @Override    protected void onResume() {        super.onResume();
        Log.d(TAG, "UserManager.sUserId=" + User.sUserId);        super.onStart();
        persistToFile();
    }    private void persistToFile() {        new Thread(new Runnable() {
            @Override            public void run() {
                User user = new User(1, "hello world", false);
                File dir = new File(MyConstants.CHAPTER_2_PATH);                if (!dir.exists()) {
                    dir.mkdirs();
                }
                File cachedFile = new File(MyConstants.CACHE_FILE_PATH);
                ObjectOutputStream objectOutputStream = null;                try {
                    objectOutputStream = new ObjectOutputStream(new FileOutputStream(cachedFile));
                    objectOutputStream.writeObject(user);
                    Log.d(TAG, "persist user:" + user);
                    mTextView.setText("persist user:" + user);
                } catch (IOException e) {
                    e.printStackTrace();
                } finally {
                    MyUtils.close(objectOutputStream);
                }
            }
        }).start();
    }
}

7.修改AndroidManifest.xml文件

<?xml version="1.0" encoding="utf-8"?><manifest 
   xmlns:android="http://schemas.android.com/apk/res/android"    package="com.zhangmiao.ipcdemo"><uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/><application        android:allowBackup="true"        android:icon="@mipmap/ic_launcher"        android:label="@string/app_name"        android:supportsRtl="true"        android:theme="@style/AppTheme"><activity            android:name=".MainActivity"            android:label="@string/app_name"            android:launchMode="standard"            android:theme="@style/AppTheme.NoActionBar"><intent-filter><action android:name="android.intent.action.MAIN"/><category android:name="android.intent.category.LAUNCHER"/></intent-filter></activity><activity            android:name=".SecondActivity"            android:configChanges="screenLayout"            android:label="@string/app_name"            android:process=":remote"/>

</application></manifest>


















本文转自xmgdc51CTO博客,原文链接:http://blog.51cto.com/12953214/1940446 ,如需转载请自行联系原作者



相关文章
|
6月前
|
Android开发 开发者
Android自定义View之不得不知道的文件attrs.xml(自定义属性)
本文详细介绍了如何通过自定义 `attrs.xml` 文件实现 Android 自定义 View 的属性配置。以一个包含 TextView 和 ImageView 的 DemoView 为例,讲解了如何使用自定义属性动态改变文字内容和控制图片显示隐藏。同时,通过设置布尔值和点击事件,实现了图片状态的切换功能。代码中展示了如何在构造函数中解析自定义属性,并通过方法 `setSetting0n` 和 `setbackeguang` 实现功能逻辑的优化与封装。此示例帮助开发者更好地理解自定义 View 的开发流程与 attrs.xml 的实际应用。
159 2
Android自定义View之不得不知道的文件attrs.xml(自定义属性)
|
6月前
|
Java Android开发
Android studio中build.gradle文件简单介绍
本文解析了Android项目中build.gradle文件的作用,包括jcenter仓库配置、模块类型定义、包名设置及依赖管理,涵盖本地、库和远程依赖的区别。
589 19
|
6月前
|
存储 XML Java
Android 文件数据储存之内部储存 + 外部储存
简介:本文详细介绍了Android内部存储与外部存储的使用方法及核心原理。内部存储位于手机内存中,默认私有,适合存储SharedPreferences、SQLite数据库等重要数据,应用卸载后数据会被清除。外部存储包括公共文件和私有文件,支持SD卡或内部不可移除存储,需申请权限访问。文章通过代码示例展示了如何保存、读取、追加、删除文件以及将图片保存到系统相册的操作,帮助开发者理解存储机制并实现相关功能。
1650 2
|
9月前
|
移动开发 安全 Java
Android历史版本与APK文件结构
通过以上内容,您可以全面了解Android的历史版本及其主要特性,同时掌握APK文件的结构和各部分的作用。这些知识对于理解Android应用的开发和发布过程非常重要,也有助于在实际开发中进行高效的应用管理和优化。希望这些内容对您的学习和工作有所帮助。
897 83
|
ARouter Android开发
Android不同module布局文件重名被覆盖
Android不同module布局文件重名被覆盖
|
Java Android开发 C++
Android Studio JNI 使用模板:c/cpp源文件的集成编译,快速上手
本文提供了一个Android Studio中JNI使用的模板,包括创建C/C++源文件、编辑CMakeLists.txt、编写JNI接口代码、配置build.gradle以及编译生成.so库的详细步骤,以帮助开发者快速上手Android平台的JNI开发和编译过程。
1006 1
|
存储 数据库 Android开发
安卓Jetpack Compose+Kotlin,支持从本地添加音频文件到播放列表,支持删除,使用ExoPlayer播放音乐
为了在UI界面添加用于添加和删除本地音乐文件的按钮,以及相关的播放功能,你需要实现以下几个步骤: 1. **集成用户选择本地音乐**:允许用户从设备中选择音乐文件。 2. **创建UI按钮**:在界面中创建添加和删除按钮。 3. **数据库功能**:使用Room数据库来存储音频文件信息。 4. **更新ViewModel**:处理添加、删除和播放音频文件的逻辑。 5. **UI实现**:在UI层支持添加、删除音乐以及播放功能。
|
存储 监控 数据库
Android经典实战之OkDownload的文件分段下载及合成原理
本文介绍了 OkDownload,一个高效的 Android 下载引擎,支持多线程下载、断点续传等功能。文章详细描述了文件分段下载及合成原理,包括任务创建、断点续传、并行下载等步骤,并展示了如何通过多种机制保证下载的稳定性和完整性。
552 1
|
开发工具 git 索引
repo sync 更新源码 android-12.0.0_r34, fatal: 不能重置索引文件至版本 ‘v2.27^0‘。
本文描述了在更新AOSP 12源码时遇到的repo同步错误,并提供了通过手动git pull更新repo工具来解决这一问题的方法。
565 1
|
ARouter Android开发
Android不同module布局文件重名被覆盖
Android不同module布局文件重名被覆盖
789 0