前些天还自已写了一个内存文件用于http数据交互使用,没想到今天翻代码,尽然找到类似的现有实现,真是汗! 呵呵,那么就在这简要介绍一下吧!
应用场合:
1、对于I/O需要频繁操作的,主要是和外部存储相关的I/O操作,MemoryFile通过将 NAND或SD卡上的文件,分段映射到内存中进行修改处理,这样就用高速的RAM代替了ROM或SD卡,性能自然提高不少,对于Android手机而言同时还减少了电量消耗。
2、对于java与jni大数据交互的话使用起来还是相当方便,只要将数据格式定义好即可。
类 MemoryFile.java
路径:frameworks\base\core\java\android\os
JNI 实现 frameworks\base\core\jni\android_os_MemoryFile.cpp
用法:MemoryFileTest.java 这个文件中有例子, 我这里列一下基本的用法函数
- <span style="font-size:16px;"> @SmallTest
- public void testFileDescriptor() throws Exception {
- MemoryFile file = new MemoryFile("MemoryFileTest", 1000000);
- MemoryFile ref = new MemoryFile(file.getFileDescriptor(), file.length(), "r");
- byte[] buffer;
- <span style="color:#000066;"> // write to original, read from reference
- file.writeBytes(testString, 0, 2000, testString.length);
- buffer = new byte[testString.length];
- ref.readBytes(buffer, 2000, 0, testString.length);
- compareBuffers(testString, buffer, testString.length);</span>
- file.close();
- ref.close(); // Doesn't actually do anything, since the file descriptor is not dup(2):ed
- }
- </span>
比较数据函数:
- <span style="font-size:16px;"> private void compareBuffers(byte[] buffer1, byte[] buffer2, int length) throws Exception {
- for (int i = 0; i < length; i++) {
- if (buffer1[i] != buffer2[i]) {
- throw new Exception("readBytes did not read back what writeBytes wrote");
- }
- }
- }</span>
ok, 使用非常简单
下面来分析下其实现的主要原素,大内存数据读写肯定离不开内存映射,这里必须使用到了 mmap / munmap
- <span style="font-size:16px;">static jint android_os_MemoryFile_mmap(JNIEnv* env, jobject clazz, jobject fileDescriptor,
- jint length, jint prot)
- {
- int fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
- <span style="color:#009900;">jint result = (jint)mmap(NULL, length, prot, MAP_SHARED, fd, 0);</span>
- if (!result)
- jniThrowException(env, "java/io/IOException", "mmap failed");
- return result;
- }
- static void android_os_MemoryFile_munmap(JNIEnv* env, jobject clazz, jint addr, jint length)
- {
- <span style="color:#009900;"> int result = munmap((void *)addr, length);</span>
- if (result < 0)
- jniThrowException(env, "java/io/IOException", "munmap failed");
- }</span>
分配内存使用匿名内存实现:
- <span style="font-size:16px;">static jobject android_os_MemoryFile_open(JNIEnv* env, jobject clazz, jstring name, jint length)
- {
- const char* namestr = (name ? env->GetStringUTFChars(name, NULL) : NULL);
- int result = ashmem_create_region(namestr, length);
- </span>
数据读和写实现:
- <span style="font-size:16px;">static jint android_os_MemoryFile_read(JNIEnv* env, jobject clazz,
- jobject fileDescriptor, jint address, jbyteArray buffer, jint srcOffset, jint destOffset,
- jint count, jboolean unpinned)
- {
- int fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
- ...
- <span style="color:#3333FF;">env->SetByteArrayRegion(buffer, destOffset, count, (const jbyte *)address + srcOffset);</span>
- return count;
- }
- static jint android_os_MemoryFile_write(JNIEnv* env, jobject clazz,
- jobject fileDescriptor, jint address, jbyteArray buffer, jint srcOffset, jint destOffset,
- jint count, jboolean unpinned)
- {
- int fd = jniGetFDFromFileDescriptor(env, fileDescriptor);
- ...
- <span style="color:#3333FF;">env->GetByteArrayRegion(buffer, srcOffset, count, (jbyte *)address + destOffset);</span>
- return count;
- }
- </span>