MemoryFile匿名共享内存使用,已拿offer

把刚刚得到的ParcelFileDescriptor传递到其他进程,这个比较简单直接用binder传就可以了

通过描述共享内存文件描述取得一个描述共享内存的MemoryFile对象,并且需要让这个MemoryFile对象指向刚刚创建的共享内存。在低版本的系统中存在一个构造函数可以直接以FileDescriptor为参数构造出一个MemoryFile对象,这样构造出来的对象刚好指向FileDescriptor描述的共享内存。但是在高版本中没有样的构造函数了。所以在这里我利用了一个取巧的方式。思路是:利用构造函数

public MemoryFile(String name, int length) throws IOException

构造一个MemoryFile对象,当然此时也创建了一块新的共享内存,但是这块共享内存不是我们需要的;调用public void close()方法关闭刚刚创建的共享内存。通过前面的操作后我们得到了一个MemoryFile对象,但是这个对象没有指向任何共享内存,所以接下来我们就需要让MemoryFile对象指向我们需要的共享内存,也就是FileDescriptor描述的那块。在MemoryFile中有一个native方法:

private static native long native_mmap(FileDescriptor fd, int length, int mode)

这个方法就是把fd描述的共享内存映射到虚拟地址空间中。所以我们可以已刚刚获得的FileDescriptor 作为参数利用反射调用这个方法:

/**

  • 打开共享内存,一般是一个地方创建了一块共享内存

  • 另一个地方持有描述这块共享内存的文件描述符,调用

  • 此方法即可获得一个描述那块共享内存的MemoryFile

  • 对象

  • @param fd 文件描述

  • @param length 共享内存的大小

  • @param mode PROT_READ = 0x1只读方式打开,

  •         PROT_WRITE = 0x2可写方式打开,
    
  •         PROT_WRITE|PROT_READ可读可写方式打开
    
  • @return MemoryFile

*/

public static MemoryFile openMemoryFile(FileDescriptor fd,int length,int mode){

MemoryFile memoryFile = null;

try {

memoryFile = new MemoryFile(“tem”,1);

memoryFile.close();

Class<?> c = MemoryFile.class;

Method native_mmap = null;

Method[] ms = c.getDeclaredMethods();

for(int i = 0;ms != null&&i<ms.length;i++){

if(ms[i].getName().equals(“native_mmap”)){

native_mmap = ms[i];

}

}

ReflectUtil.setField(“android.os.MemoryFile”, memoryFile, “mFD”, fd);

ReflectUtil.setField(“android.os.MemoryFile”,memoryFile,“mLength”,length);

long address = (long) ReflectUtil.invokeMethod( null, native_mmap, fd, length, mode);

ReflectUtil.setField(“android.os.MemoryFile”, memoryFile, “mAddress”, address);

} catch (Exception e) {

e.printStackTrace();

}

return memoryFile;

}

这样我们就得到了一个指向一开始我们创建的那块共享内存的MemoryFile了,接下来就可以调用它的public int readBytes(byte[] buffer, int srcOffset, int destOffset, int count)和public void writeBytes(byte[] buffer, int srcOffset, int destOffset, int count)从共享内存中读数据和往共享内存中写数据了

最后上完整的代码:

package wzr.com.slidefinish.util;

import android.os.MemoryFile;

import android.os.ParcelFileDescriptor;

import java.io.FileDescriptor;

import java.io.IOException;

import java.lang.reflect.Method;

/**

  • 对memoryFile类的扩展

  • 1.从memoryFile对象中获取FileDescriptor,ParcelFileDescriptor

  • 2.根据一个FileDescriptor和文件length实例化memoryFile对象

*/

public class MemoryFileHelper {

/**

  • 创建共享内存对象

  • @param name 描述共享内存文件名称

  • @param length 用于指定创建多大的共享内存对象

  • @return MemoryFile 描述共享内存对象

*/

public static MemoryFile createMemoryFile(String name,int length){

try {

return new MemoryFile(name,length);

} catch (IOException e) {

e.printStackTrace();

}

return null;

}

public static MemoryFile openMemoryFile(ParcelFileDescriptor pfd,int length,int mode){

if(pfd == null){

throw new IllegalArgumentException(“ParcelFileDescriptor 不能为空”);

}

FileDescriptor fd = pfd.getFileDescriptor();

return openMemoryFile(fd,length,mode);

}

/**

  • 打开共享内存,一般是一个地方创建了一块共享内存

  • 另一个地方持有描述这块共享内存的文件描述符,调用

  • 此方法即可获得一个描述那块共享内存的MemoryFile

  • 对象

  • @param fd 文件描述

  • @param length 共享内存的大小

  • @param mode PROT_READ = 0x1只读方式打开,

  •         PROT_WRITE = 0x2可写方式打开,
    
  •         PROT_WRITE|PROT_READ可读可写方式打开
    
  • @return MemoryFile

*/

public static MemoryFile openMemoryFile(FileDescriptor fd,int length,int mode){

MemoryFile memoryFile = null;

try {

memoryFile = new MemoryFile(“tem”,1);

memoryFile.close();

Class<?> c = MemoryFile.class;

Method native_mmap = null;

Method[] ms = c.getDeclaredMethods();

for(int i = 0;ms != null&&i<ms.length;i++){

if(ms[i].getName().equals(“native_mmap”)){

native_mmap = ms[i];

}

}

ReflectUtil.setField(“android.os.MemoryFile”, memoryFile, “mFD”, fd);

ReflectUtil.setField(“android.os.MemoryFile”,memoryFile,“mLength”,length);

long address = (long) ReflectUtil.invokeMethod( null, native_mmap, fd, length, mode);

ReflectUtil.setField(“android.os.MemoryFile”, memoryFile, “mAddress”, address);

} catch (Exception e) {

e.printStackTrace();

}

return memoryFile;

}

/**

  • 获取memoryFile的ParcelFileDescriptor

  • @param memoryFile 描述一块共享内存

  • @return ParcelFileDescriptor

*/

public static ParcelFileDescriptor getParcelFileDescriptor(MemoryFile memoryFile){

if(memoryFile == null){

throw new IllegalArgumentException(“memoryFile 不能为空”);

}

ParcelFileDescriptor pfd;

FileDescriptor fd = getFileDescriptor(memoryFile);

pfd = (ParcelFileDescriptor) ReflectUtil.getInstance(“android.os.ParcelFileDescriptor”,fd);

return pfd;

}

/**

  • 获取memoryFile的FileDescriptor

  • @param memoryFile 描述一块共享内存

  • @return 这块共享内存对应的文件描述符

*/

public static FileDescriptor getFileDescriptor(MemoryFile memoryFile){

if(memoryFile == null){

throw new IllegalArgumentException(“memoryFile 不能为空”);

}

FileDescriptor fd;

fd = (FileDescriptor) ReflectUtil.invoke(“android.os.MemoryFile”,memoryFile,“getFileDescriptor”);

return fd;

}

}

ReflectUtil.java

package wzr.com.slidefinish.util;

import java.lang.reflect.Constructor;

import java.lang.reflect.Field;

import java.lang.reflect.Method;

/**

  • 反射工具类

  • Created by wuzr on 2016/6/27.

*/

public class ReflectUtil {

/**

*根据类名,参数实例化对象

  • @param className 类的路径全名

  • @param params 构造函数需要的参数

  • @return 返回T类型的一个对象

*/

public static Object getInstance(String className,Object … params){

if(className == null || className.equals(“”)){

throw new IllegalArgumentException(“className 不能为空”);

}

try {

Class<?> c = Class.forName(className);

if(params != null){

int plength = params.length;

Class[] paramsTypes = new Class[plength];

for (int i = 0; i < plength; i++) {

paramsTypes[i] = params[i].getClass();

}

Constructor constructor = c.getDeclaredConstructor(paramsTypes);

constructor.setAccessible(true);

return constructor.newInstance(params);

}

Constructor constructor = c.getDeclaredConstructor();

constructor.setAccessible(true);

return constructor.newInstance();

} catch (Exception e) {

e.printStackTrace();

}

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

【附】相关架构及资料

源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,和技术大牛一起讨论交流解决问题。

image

ac81d625c.png)

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
[外链图片转存中…(img-b0fpuiuy-1711780373226)]

【附】相关架构及资料

源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,和技术大牛一起讨论交流解决问题。

[外链图片转存中…(img-KasutooC-1711780373227)]

本文已被CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录

  • 10
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值