文件共享是一种非常不错的IPC方式,即两个进程可以通过读/写同一个文件来交换数据。和Windows系统不同,Android系统是基于Linux的,这使得并发读/写文件的操作可以没有限制地进行,甚至两个线程同时对一个文件进行读/写也是可以的(尽管这样可能会出问题)。

  使用文件共享的方式实现IPC时,文件中除了可以存储一些文本信息外,我们也可以序列化一个对象到文件系统中,然后在另一个进程中恢复这个对象。

  下面用一个例子来演示在文件系统中读/写对象的功能。

  我们在一个Module中创建两个Activity,分别是MainActivity和SecondActivity,Manifest文件中的代码如下:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="my.itgungnir.ipc"> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> <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"></activity>
<activity
android:name=".SecondActivity"
android:process=":remote">
<intent-filter>
<action android:name="android.intent.action.MAIN"/> <category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
</application> </manifest>

  我们通过在SecondActivity中设置 android:process 属性,让SecondActivity运行在另一个进程中,然后通过控制 <intent-filter> 标签的位置来决定当前启动哪个Activity(即哪个进程)。

  另外,不要忘记在Manifest文件中添加访问SD卡的权限。

  在这个例子中,我们将一个User对象存储到文件中,我们首先需要对User类进行序列化,以保证能够在进程中传递,User类中的代码如下:

public class User implements Serializable {
private static final long serialVersionUID = 1L; public int userId;
public String userName;
public boolean isMale; public User(int userId, String userName, boolean isMale) {
this.userId = userId;
this.userName = userName;
this.isMale = isMale;
}
}

  需要注意的是,将对象存储到文件系统中的过程可以理解为持久化的过程,而Parcelable接口不适合用来序列化可持久化的数据,因此这个我们必须使用Serializable接口进行序列化。

  在MainActivity中有一个按钮,当点击这个按钮的时候,就会生成一个User类的对象,存储到设备的SD卡中,代码如下:

public class MainActivity extends AppCompatActivity implements View.OnClickListener {
private Button btn; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btn = (Button) findViewById(R.id.a_btn);
btn.setOnClickListener(this);
} @Override
public void onClick(View v) {
ActivityCompat.requestPermissions(MainActivity.this, new String[]{android.Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);
} @Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
new Thread(new Runnable() {
@Override
public void run() {
User user = new User(1001, "Alice", false);
String sdCardState = Environment.getExternalStorageState();
if (Environment.MEDIA_MOUNTED.equals(sdCardState)) {
File dir = new File(Environment.getExternalStorageDirectory().getAbsolutePath() + "/file_ipc/");
if (!dir.exists()) {
dir.mkdirs();
}
File tmpFile = new File(dir + File.separator + "fileipc.txt");
if (tmpFile.exists()) {
tmpFile.delete();
}
File file = new File(dir + File.separator + "fileipc.txt");
if (!file.exists()) {
try {
file.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
}
ObjectOutputStream oos = null;
try {
oos = new ObjectOutputStream(new FileOutputStream(file));
oos.writeObject(user);
} catch (IOException e) {
e.printStackTrace();
} finally {
if (oos != null) {
try {
oos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
}).start();
}
}

  需要说明的是,笔者做这个DEMO使用的是Android 6.0的SDK,因此在涉及到权限的时候,除了在Manifest文件中声明权限之外,还在在Activity中通过 ActivityCompat.requestPermissions() 方法申请权限,然后在 onRequestPermissionsResult() 方法中回调要执行的代码。

  通过运行项目(此时 <intent-filter> 标签在MainActivity下),User对象就被存储到SD卡的文件系统中了。

  SecondActivity中有一个TextView,用来显示从SD卡的文件系统中读取出来的User对象中的数据。SecondActivity中的代码如下:

public class SecondActivity extends AppCompatActivity {
private TextView tv; private Handler textHandler; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
tv = (TextView) findViewById(R.id.b_tv);
initHandler();
initView();
} private void initHandler() {
textHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
if (msg.what == 1) {
User user = (User) msg.obj;
tv.setText("读取文件成功!\r\n");
tv.append("用户编号:" + user.userId + "\r\n");
tv.append("用户姓名:" + user.userName + "\r\n");
tv.append("用户性别:" + (user.isMale ? "男" : "女") + "\r\n");
}
}
};
} private void initView() {
ActivityCompat.requestPermissions(SecondActivity.this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, 1);
} @Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
new Thread(new Runnable() {
@Override
public void run() {
User user = null;
File file = new File(Environment.getExternalStorageDirectory().getAbsolutePath() + File.separator + "file_ipc" + File.separator + "fileipc.txt");
if (file.exists()) {
ObjectInputStream ois = null;
try {
ois = new ObjectInputStream(new FileInputStream(file));
user = (User) ois.readObject();
Message msg = Message.obtain();
msg.what = 1;
msg.obj = user;
textHandler.sendMessage(msg);
} catch (IOException | ClassNotFoundException e) {
e.printStackTrace();
} finally {
if (ois != null) {
try {
ois.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
}).start();
}
}

  注意:由于我们的读写操作都是耗时操作,因此我们新开了一个现成执行这些代码,因此当我们需要将读取到的数据展示到TextView中的时候,就需要使用Handler将数据传送到主线程中,再跟新UI界面。

  在Manifest文件中将 <intent-filter> 标签移动到SecondActivity下,再次运行项目,就可以看到SD卡中存储的User对象中的信息就被展示到TextView中了。

  最后,注意一点,SharedPreferences虽然也是以文件的形式存储数据的,但是不适合在IPC中使用,原因是Android系统对SharedPreferences的读/写有一定的缓存策略,即在内存中会有一份SharedPreferences文件的缓存,因此在多进程的模式下,系统对SharedPreferences的读写就变得不可靠,当面对高并发的读/写访问时,SharedPreferences有很大几率会丢失数据,因此,不建议在IPC中使用SharedPreferences。

【Android - IPC】之使用文件共享实现IPC的更多相关文章

  1. android service 的各种用法(IPC、AIDL)

    http://my.oschina.net/mopidick/blog/132325 最近在学android service,感觉终于把service的各种使用场景和用到的技术整理得比较明白了,受益颇 ...

  2. Android AIDL 进行进程间通讯(IPC)

    编写AIDL文件时,需要注意: 1.接口名和aidl文件名相同. 2.接口和方法前不用加访问权限修饰符 (public.private.protected等,也不能用final.static). 3. ...

  3. Android AIDL使用详解_Android IPC 机制详解

    一.概述 AIDL 意思即 Android Interface Definition Language,翻译过来就是Android接口定义语言,是用于定义服务器和客户端通信接口的一种描述语言,可以拿来 ...

  4. Linux IPC实践(13) --System V IPC综合实践

    实践:实现一个先进先出的共享内存shmfifo 使用消息队列即可实现消息的先进先出(FIFO), 但是使用共享内存实现消息的先进先出则更加快速; 我们首先完成C语言版本的shmfifo(基于过程调用) ...

  5. Android IPC机制全解析<一>

    概要 多进程概念及多进程常见注意事项 IPC基础:Android序列化和Binder 跨进程常见的几种通信方式:Bundle通过Intent传递数据,文件共享,ContentProvider,基于Bi ...

  6. Android IPC机制基础

    概要 多进程概念及多进程常见注意事项 IPC基础:Android序列化和Binder 跨进程常见的几种通信方式:Bundle通过Intent传递数据,文件共享,ContentProvider,基于Bi ...

  7. Android进阶笔记18:选用合适的IPC方式

    1. 相信大家都知道Android进程间通信方式很多,比如AIDL.Messenger等等,接下来我就总结一下这些IPC方式优缺点. 2. IPC方式的优缺点和适用场景 3. 附加:使用Intent实 ...

  8. [置顶] 深入理解android之IPC机制与Binder框架

    [android之IPC机制与Binder框架] [Binder框架.Parcel.Proxy-Stub以及AIDL] Abstract [每个平台都会有自己一套跨进程的IPC机制,让不同进程里的两个 ...

  9. Android IPC 结篇

    一.概述 Android 的 IPC 方式有 Bundle .共享文件.AIDL .Messenger .ContentProvider .Socket ,我们在实现进程间通信时要选择哪一种方式来实现 ...

随机推荐

  1. Shiro笔记---身份验证

    1.shiro有哪些主要功能 2.搭建shiro环境(*) idea2018.2.maven3.5.4.jdk1.8   项目结构: pom.xml: <dependencies> < ...

  2. (八十六)c#Winform自定义控件-表格优化

    出处:http://www.hzhcontrols.com/原文:http://www.hzhcontrols.com/blog-149.html本文版权归www.hzhcontrols.com所有欢 ...

  3. vsftpd超实用技巧详解

    简介: vsftpd是"very secure FTP daemon"的缩写,是一个完全免费的.开放源代码的ftp服务器软件. 工作原理: vsftpd使用ftp协议,该协议属于应 ...

  4. CentOS生产环境无网络安装percona-xtrabackup2.4【RPM安装教程】

    Percona XtraBackup 8.0不支持对在MySQL 8.0之前的版本,Percona Server for MySQL或 Percona XtraDB Cluster中创建的数据库进行备 ...

  5. deepin扬声器/耳机没有声音解决方案

    昨天准备在deepin系统下看视频学习一下Linux,结果登入deepin系统后发现不论是外放还是插耳机竟然都没有声音,这种情况以前也出现过,只不过没有在意,后来就自己又好了,今天这次可真是让我决定要 ...

  6. 【XSY2525】Maze 2017多校

    Description 考虑一个 N×M 的网格,每个网格要么是空的,要么是障碍物.整个网格四周都是墙壁(即第1行和第n行,第1列和第m列都是墙壁),墙壁有且仅有两处开口,分别代表起点和终点.起点总是 ...

  7. Go defer使用

    defer使用语法 //defer后面必须是函数调用语句或方法调用语句,不能是其他语句,否则编译器会出错. package main import ( "fmt" ) func f ...

  8. centos下docker离线部署

    安装准备 Docker可以让开发者打包他们的应用以及依赖包到一个轻量级.可移植的容器中,然后发布到任何流行的Linux机器上,也可以实现虚拟化. 环境要求 Centos 安装包下载地址 安装包下载以下 ...

  9. jquery layui的巨坑

    jquery layui的巨坑 layui 模块不能写在ajax里 因为 layui只能执行一次 第二次会没效果 再执行需要刷新页面再执行

  10. win7设置docker默认服务端地址

    目录 win7设置docker默认服务端地址 1.开启docker远程访问 2.本地调整 2.1 docker.exe重命名 2.2 添加docker.bat 2.3 添加快速切换功能 3.使用验证 ...