Android使用文件进行IPC

发布时间 - 2026-01-10 22:11:26    点击率:

一、文件进行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>

完整代码下载地址:https://github.com/ZhangMiao147/IPCDemo

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# Android  # 文件  # IPC  # Android IPC进程间通信详解最新AndroidStudio的AIDL操作)  # Android系统进程间通信(IPC)机制Binder中的Client获得Server远程接口过程源  # Android系统进程间通信(IPC)机制Binder中的Server启动过程源代码分析  # Android系统进程间通信(IPC)机制Binder中的Server和Client获得Servic  # 浅谈Service Manager成为Android进程间通信(IPC)机制Binder守护进程之路  # Android进程间通信(IPC)机制Binder简要介绍  # Android IPC机制利用Messenger实现跨进程通信  # Android IPC机制绑定Service实现本地通信  # android IPC之binder通信机制  # 开放IPC$共享  # 序列化  # 很好  # 还可以  # 将会  # 下载地址  # 对其  # 使其  # 这可  # 文件系统  # 大家多多  # 卡上  # finally  # close  # MyUtils  # catch  # IOException  # printStackTrace  # objectInputStream 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: laravel怎么在请求结束后执行任务(Terminable Middleware)_laravel Terminable Middleware请求结束任务执行方法  laravel怎么用DB facade执行原生SQL查询_laravel DB facade原生SQL执行方法  EditPlus 正则表达式 实战(3)  使用PHP下载CSS文件中的所有图片【几行代码即可实现】  Win11关机界面怎么改_Win11自定义关机画面设置【工具】  微信小程序 require机制详解及实例代码  Laravel storage目录权限问题_Laravel文件写入权限设置  原生JS实现图片轮播切换效果  如何快速打造个性化非模板自助建站?  Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】  Laravel如何使用Facades(门面)及其工作原理_Laravel门面模式与底层机制  laravel怎么使用数据库工厂(Factory)生成带有关联模型的数据_laravel Factory生成关联数据方法  Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】  Python数据仓库与ETL构建实战_Airflow调度流程详解  Laravel怎么设置路由分组Prefix_Laravel多级路由嵌套与命名空间隔离【步骤】  Laravel怎么做数据加密_Laravel内置Crypt门面的加密与解密功能  Laravel怎么处理异常_Laravel自定义异常处理与错误页面教程  如何用AI帮你把自己的生活经历写成一个有趣的故事?  敲碗10年!Mac系列传将迎来「触控与联网」双革新  Laravel如何使用Eloquent ORM进行数据库操作?(CRUD示例)  进行网站优化必须要坚持的四大原则  laravel怎么实现图片的压缩和裁剪_laravel图片压缩与裁剪方法  Laravel用户密码怎么加密_Laravel Hash门面使用教程  在Oracle关闭情况下如何修改spfile的参数  如何快速搭建高效服务器建站系统?  如何在腾讯云服务器上快速搭建个人网站?  Laravel如何创建自定义Artisan命令?(代码示例)  如何用AWS免费套餐快速搭建高效网站?  Laravel定时任务怎么设置_Laravel Crontab调度器配置  Laravel的Blade指令怎么自定义_创建你自己的Laravel Blade Directives  网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?  公司网站制作价格怎么算,公司办个官网需要多少钱?  如何在IIS管理器中快速创建并配置网站?  宙斯浏览器视频悬浮窗怎么开启 边看视频边操作其他应用教程  如何使用 Go 正则表达式精准提取括号内首个纯字母标识符(忽略数字与嵌套)  大连 网站制作,大连天途有线官网?  php读取心率传感器数据怎么弄_php获取max30100的心率值【指南】  实现点击下箭头变上箭头来回切换的两种方法【推荐】  Laravel如何实现API版本控制_Laravel API版本化路由设计策略  用yum安装MySQLdb模块的步骤方法  如何用狗爹虚拟主机快速搭建网站?  Laravel Vite是做什么的_Laravel前端资源打包工具Vite配置与使用  网站优化排名时,需要考虑哪些问题呢?  清除minerd进程的简单方法  如何在阿里云完成域名注册与建站?  如何快速上传建站程序避免常见错误?  大连企业网站制作公司,大连2025企业社保缴费网上缴费流程?  制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?  C++用Dijkstra(迪杰斯特拉)算法求最短路径  Laravel如何使用Livewire构建动态组件?(入门代码)