设为首页收藏本站
网站公告 | 这是第一条公告
     

 找回密码
 立即注册
缓存时间23 现在时间23 缓存数据 荣耀也罢,屈辱也罢,都要以平和的心态去面对,少一些无奈与感慨,多一份从容和淡然。晚安!

荣耀也罢,屈辱也罢,都要以平和的心态去面对,少一些无奈与感慨,多一份从容和淡然。晚安!

查看: 824|回复: 2

Android实现多进程并发控制的两种方案

[复制链接]

  离线 

TA的专栏

  • 打卡等级:热心大叔
  • 打卡总天数:238
  • 打卡月天数:1
  • 打卡总奖励:3564
  • 最近打卡:2025-12-16 10:26:39
等级头衔

等級:晓枫资讯-上等兵

在线时间
0 小时

积分成就
威望
0
贡献
412
主题
360
精华
0
金钱
4751
积分
855
注册时间
2023-1-6
最后登录
2025-12-16

发表于 2025-9-8 13:08:52 | 显示全部楼层 |阅读模式
一、问题背景

当一个App中存在多个进程时例如存在 主进程,辅进程两个进程,两个进程都会去向A文件中写入数据。但是我们业务中希望每次仅允许有一个进程向A文件写入内容。即当主进程写入时,辅进程要等待主进程写完之后才可以写入,防止出现并发修改导致数据异常的问题。
在实际的场景上,例如在我们的项目中未使用MMKV之前,KV存储是自行实现的多进程并发的SP。

二、实现方案


1、方案1:仅一个进程负责写

将所有的写入操作调整到同一个进程中,这样就相当于规避了多进程并发问题。
我们可以通过提供一个ContantProvider或者Service来是实现这个功能。
以下是使用ContentProvider的方式:

FileProvider
  1. public class FileProvider extends ContentProvider {
  2.     private static final String AUTHORITY = "com.example.fileprovider";
  3.     private static final Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY + "/file");

  4.     // 文件锁,确保单进程写入
  5.     private static final Object fileLock = new Object();

  6.     @Override
  7.     public boolean onCreate() {
  8.         return true;
  9.     }

  10.     @Override
  11.     public Cursor query(Uri uri, String[] projection,
  12.         String selection, String[] selectionArgs, String sortOrder) {
  13.         return null; // 不提供查询功能
  14.     }

  15.     @Override
  16.     public String getType(Uri uri) {
  17.         return null;
  18.     }

  19.     @Override
  20.     public Uri insert(Uri uri, ContentValues values) {
  21.         return null; // 不提供插入功能
  22.     }

  23.     @Override
  24.     public int delete(Uri uri, String selection, String[] selectionArgs) {
  25.         return 0; // 不提供删除功能
  26.     }

  27.     @Override
  28.     public int update(Uri uri, ContentValues values,
  29.         String selection, String[] selectionArgs) {
  30.         return 0; // 不提供更新功能
  31.     }

  32.     // 自定义方法:写入文件
  33.     @Override
  34.     public Bundle call(String method, String arg, Bundle extras) {
  35.         if ("writeToFile".equals(method)) {
  36.             String content = extras.getString("content");
  37.             synchronized (fileLock) {
  38.                 writeToFile(content);
  39.             }
  40.             Bundle result = new Bundle();
  41.             result.putBoolean("success", true);
  42.             return result;
  43.         }
  44.         return super.call(method, arg, extras);
  45.     }

  46.     // 实际写入文件的逻辑
  47.     private void writeToFile(String content) {
  48.         File file = new File(getContext().getFilesDir(), "A.txt");
  49.         try (FileOutputStream fos = new FileOutputStream(file, true)) {
  50.             fos.write(content.getBytes());
  51.         } catch (IOException e) {
  52.             e.printStackTrace();
  53.         }
  54.     }
  55. }
复制代码
注册
  1. <provider
  2.     android:name=".FileProvider"
  3.     android:authorities="com.example.fileprovider"
  4.     android:exported="true" />
复制代码
写入逻辑
  1.     private void writeToFileViaProvider(String content) {
  2.         Uri uri = Uri.parse("content://com.example.fileprovider/file");
  3.         ContentResolver resolver = getContentResolver();
  4.         
  5.         Bundle extras = new Bundle();
  6.         extras.putString("content", content);
  7.         
  8.         try {
  9.             Bundle result = resolver.call(uri, "writeToFile", null, extras);
  10.             if (result != null && result.getBoolean("success")) {
  11.                 Log.d("FileProvider", "Write successful");
  12.             }
  13.         } catch (Exception e) {
  14.             Log.e("FileProvider", "Failed to write file", e);
  15.         }
  16.     }
复制代码
使用Service + Binder的方式,代码比较简单,这里就不写了。

2、方案2:通过文件锁的方式

文件锁主要是利用FileChannel、FileLock来控制多进程并发。

关于 Channel

Channel 经常翻译为通道,类似 IO 中的流,用于读取和写入。不用像BIO那样,读数据和写数据需要不同的数据通道。
  1. public interface Channel extends Closeable {

  2.     /**
  3.      * Tells whether or not this channel is open.
  4.      *
  5.      * @return <tt>true</tt> if, and only if, this channel is open
  6.      */
  7.     public boolean isOpen();

  8.     /**
  9.      * Closes this channel.
  10.      */
  11.     public void close() throws IOException;

  12. }
复制代码
我们常用的Channel有:

  • FileChannel:文件通道,用于文件的读和写。
  • DatagramChannel:用于UDP连接的接收和发送。
  • SocketChannel:把它理解为TCP连接通道,简单理解就是TCP客户端。
  • ServerSocketChannel:TCP对应的服务端,用于监听某个端口进来的请求。

FileChannel

FileChannel 是 Java NIO (New I/O) 中的一个类,用于对文件进行高效的读写操作。它提供了比传统
  1. FileInputStream
复制代码
  1. FileOutputStream
复制代码
更灵活和高效的文件操作方式。
所有的NIO操作始于通道,通道是数据来源或数据写入的目的地。其与 Buffer 打交道,读操作的时候将 Channel 中的数据填充到 Buffer 中,而写操作时将 Buffer 中的数据写入到 Channel 中。
FileChannel的获取方式:
通过FileInputStream/FileOutputStream
  1. // 通过 FileInputStream/FileOutputStream (只读或只写)
  2. FileInputStream fis = new FileInputStream("file.txt");
  3. FileChannel readChannel = fis.getChannel();

  4. FileOutputStream fos = new FileOutputStream("file.txt");
  5. FileChannel writeChannel = fos.getChannel();
复制代码
通过RandomAccessFile
  1. // 通过 RandomAccessFile
  2. RandomAccessFile raf = new RandomAccessFile("file.txt", "rw");
  3. FileChannel channel = raf.getChannel();
复制代码
通过FileChannel.open()
  1. FileChannel channel = FileChannel.open(Paths.get("file.txt"), StandardOpenOption.READ);
复制代码
在我们示例代码中选择了使用
  1. FileOutputStream
复制代码
来获取FileChannel。

FileLock
  1. FileLock
复制代码
表示文件或文件区域的锁,用于控制多个进程或线程对同一文件的并发访问。
锁的类型

  • 共享锁 (Shared Lock) :多个进程可同时持有,用于读操作
  • 排他锁 (Exclusive Lock) :一次只能由一个进程持有,用于写操作
通过文件锁的方式控制多进程并发的 示例代码:
  1. public class FileWriter {

  2.     private static final String FILE_PATH = "/path/to/your/file.txt";

  3.     public void writeToFile(String content) {
  4.         File file = new File(FILE_PATH);
  5.         try {
  6.             FileOutputStream fos = new FileOutputStream(file, true);
  7.             FileChannel channel = fos.getChannel())
  8.             // 获取独占锁
  9.             FileLock lock = channel.lock();
  10.             try {
  11.                 // 写入文件
  12.                 fos.write(content.getBytes());
  13.             } finally {
  14.                 // 释放锁
  15.                 lock.release();
  16.             }
  17.         } catch (IOException e) {
  18.             e.printStackTrace();
  19.         }
  20.     }
  21. }
复制代码
三、总结

以上简单介绍了一下两种控制多进程并发的方案。
其中使用ContentProvider或者Service的方式将所有的操作控制在同一个进程中的方案逻辑清晰,但是代码量比较多。尤其是使用Service的方式,虽然上面我们每个给出示例代码,但是可以想象没新增一个进程都需要写相关的代码,写起来就比较啰嗦了。
而ContentProvicer的方式,系统中也有很多相关的实现方案,例如更新媒体文件,更新联系人数据等。
使用文件锁的方式对于仅熟悉Android不熟悉Java的同学不容易想到,所以本篇也同时简单介绍了一下FileChannel以及FileLock。
到此这篇关于Android实现多进程并发控制的两种方案的文章就介绍到这了,更多相关Android多进程并发控制内容请搜索晓枫资讯以前的文章或继续浏览下面的相关文章希望大家以后多多支持晓枫资讯!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
晓枫资讯-科技资讯社区-免责声明
免责声明:以上内容为本网站转自其它媒体,相关信息仅为传递更多信息之目的,不代表本网观点,亦不代表本网站赞同其观点或证实其内容的真实性。
      1、注册用户在本社区发表、转载的任何作品仅代表其个人观点,不代表本社区认同其观点。
      2、管理员及版主有权在不事先通知或不经作者准许的情况下删除其在本社区所发表的文章。
      3、本社区的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,举报反馈:点击这里给我发消息进行删除处理。
      4、本社区一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
      5、以上声明内容的最终解释权归《晓枫资讯-科技资讯社区》所有。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~

  离线 

TA的专栏

等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
0
主题
0
精华
0
金钱
11
积分
2
注册时间
2024-8-20
最后登录
2024-8-20

发表于 2025-11-12 08:54:22 | 显示全部楼层
顶顶更健康!!!
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~

  离线 

TA的专栏

等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
0
主题
0
精华
0
金钱
19
积分
18
注册时间
2022-12-24
最后登录
2022-12-24

发表于 2025-11-15 18:29:48 | 显示全部楼层
感谢楼主,顶。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~
严禁发布广告,淫秽、色情、赌博、暴力、凶杀、恐怖、间谍及其他违反国家法律法规的内容。!晓枫资讯-社区
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

1楼
2楼
3楼

手机版|晓枫资讯--科技资讯社区 本站已运行

CopyRight © 2022-2025 晓枫资讯--科技资讯社区 ( BBS.yzwlo.com ) . All Rights Reserved .

晓枫资讯--科技资讯社区

本站内容由用户自主分享和转载自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

如有侵权、违反国家法律政策行为,请联系我们,我们会第一时间及时清除和处理! 举报反馈邮箱:点击这里给我发消息

Powered by Discuz! X3.5

快速回复 返回顶部 返回列表