一尘不染

使用malloc分配的内存超过现有内存

linux

每次从stdin读取字母“ u”时,此代码段将分配2Gb,并且在读取“ a”后将初始化所有分配的字符。

#include <iostream>
#include <stdlib.h>
#include <stdio.h>
#include <vector>
#define bytes 2147483648
using namespace std;
int main()
{
    char input [1];
    vector<char *> activate;
    while(input[0] != 'q')
    {
        gets (input);
        if(input[0] == 'u')
        {
            char *m = (char*)malloc(bytes);
            if(m == NULL) cout << "cant allocate mem" << endl;
            else cout << "ok" << endl;
            activate.push_back(m);
        }
        else if(input[0] == 'a')
        {
            for(int x = 0; x < activate.size(); x++)
            {
                char *m;
                m = activate[x];
                for(unsigned x = 0; x < bytes; x++)
                {
                    m[x] = 'a';
                }
            }
        }
    }
    return 0;
}

我在具有3Gb内存的linux虚拟机上运行此代码。在使用htop工具监视系统资源使用情况时,我已经意识到malloc操作不会反映在资源上。

例如,当我仅输入一次“ u”(即分配2GB的堆内存)时,我看不到htop中的内存使用量增加2GB。只有当我输入“
a”(即初始化)时,我才会看到内存使用量增加。

结果,我能够“分配”更多的堆内存。例如,我可以malloc
6GB(比我的ram和swap内存大),而malloc会允许它(即malloc不返回NULL)。但是,当我尝试初始化分配的内存时,我可以看到内存和交换内存已满,直到进程被杀死为止。

-我的问题:

1.这是内核错误吗?

2.有人可以向我解释为什么允许这种行为吗?


阅读 506

收藏
2020-06-02

共1个答案

一尘不染

这称为 内存过量使用 。您可以通过以超级用户身份运行来禁用它:

 echo 2 > /proc/sys/vm/overcommit_memory

而且它不是我喜欢的内核功能(因此我总是禁用它)。参见malloc(3)mmap(2)proc(5)

注意:echo 0不是echo 2 经常 -但不是总是-也可以。阅读文档(特别proc是我刚刚链接到的手册页)。

2020-06-02