这是我正在使用的代码的简化版本
java :
private native void malloc(int bytes);
private native void free();
// this is called when I want to create a very large buffer in native memory
malloc(32 * 1024 * 1024);
// EDIT: after allocating, we need to initialize it before Android sees it as anythign other than a "reservation"
memset(blob, '\0', sizeof(char) * bytes);
...
// and when I'm done, I call this
free()
C:
static char* blob = NULL;
void Java_com_example_MyClass_malloc(JNIEnv * env, jobject this, jint bytes)
{
blob = (char*) malloc(sizeof(char) * bytes);
if (NULL == blob) {
__android_log_print(ANDROID_LOG_DEBUG, DEBUG_TAG, "Failed to allocate memory\n");
} else {
char m[50];
sprintf(m, "Allocated %d bytes", sizeof(char) * bytes);
__android_log_print(ANDROID_LOG_DEBUG, DEBUG_TAG, m);
}
}
void Java_com_example_MyClass_free(JNIEnv * env, jobject this)
{
free(blob);
blob = NULL;
}
现在,当我从 MyClass.java 调用 malloc() 时,我希望看到分配了 32M 的内存,并且我将能够在某处观察到可用内存的下降。
然而,我在
adb shell dumpsys meminfo
或 adb shell cat /proc/meminfo
中都没有看到任何迹象。我对 C 很陌生,但有很多 Java 经验。我希望在 Dalvik 的堆之外分配一堆内存(因此它不由 Android/dalvik 管理)用于测试目的。 Hackbod 让我相信 Android 目前没有对 Native 代码中分配的内存量设置限制,所以这似乎是正确的方法。我这样做对吗? 最佳答案
在 memset()
之后,您应该会看到“私有(private)/脏”页面的增加。如果您在设备上安装了额外的开发人员命令行实用程序,您可以运行 procrank
或 showmap <pid>
来轻松查看。需要有root权限的设备。
如果失败,让进程在分配前后将 /proc/self/maps
的内容复制到文件中。 (最简单的方法是将其写入外部存储;您将需要 list 中的 WRITE_EXTERNAL_STORAGE
权限。)通过比较 map 输出,您应该看到一个新的 32MB 区域,或一个现有区域扩展了 32MB。这是有效的,因为 32MB 高于 dlmalloc 的内部堆阈值,因此应该使用对 mmap() 的调用来分配内存。
您可以从 native 代码分配的内存量没有固定限制。然而,你分配的越多,你就越能看到内核的低内存进程杀手。
关于Android NDK : Why is this malloc() having no observable effect?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/15956481/