解决linux下出现 Cannot allocate memory 的问题

说明:
Linux下有个内核参数overcommit_memory,是内存分配策略,程序在启动的时候会先去申请内存。
overcommit_memory此参数决定是否接受超大内存请求的条件。这个参数有三个可能的值:

0:默认设置。内核执行启发式内存过量使用处理,方法是估算可用内存量,并拒绝明显无效的请求。
   遗憾的是因为内存是使用启发式而非准确算法计算进行部署,这个设置有时可能会造成系统中的可用内存超载。
1:内核执行无内存过量使用处理。使用这个设置会增大内存超载的可能性,但也可以增强大量使用内存任务的性能。
2:内存拒绝等于或者大于总可用 swap 大小以及overcommit_ratio指定的物理RAM比例的内存请求。
   如果您希望减小内存过度使用的风险,这个设置就是最好的。

目前的内存申请和可用情况cat /proc/meminfo | grep Commit
CommitLimit:表示系统可申请的总内存
Committed_AS:为当前已经申请的内存

方法一:(不推荐)
将vm.overcommit_memory设为1即可。
方式1、编辑/etc/sysctl.conf ,改vm.overcommit_memory=1,然后sysctl -p使配置文件生效
方式2、sysctl vm.overcommit_memory=1
方式3、echo 1 > /proc/sys/vm/overcommit_memory,然后sysctl -p永久生效,如果没有/etc/sysctl.conf,则每次开机都需要设置一次

方法二:

使用vfork函数

fork与vfork的区别
1、fork要拷贝父进程的数据段,而vfork则不需要完全拷贝父进程的数据段,在子进程没有调用exec和exit之前,子进程与父进程共享数据段
2、fork不对父子进程的执行次序进行任何限制,而在vfork调用中,子进程先运行,父进程挂起(子进程占用父进程地址空间,即相当于现成的概念,
父进程阻塞),直到子进程调用了exec或exit之后,父子进程的执行次序才不再有限制,这个地方要注意可能引起死锁。
3、结束子进程不用exit(0),而使用_exit(0)。  
这是因为_exit(0)在结束进程时,不对标准I/O流进行任何操作。而exit(0)则会关闭进程的所有标准I/O流。
vfork的作用
函数vfork的作用是创建一个子进程,而这个子进程的作用是用于调用exec()函数族,从而再执行一个新进程(往往是用来执行其它的程序)
代码空间
在程序的执行效果上,fork会将父进程的地址空间复制一份;但是vfork并不是这么做,而是vfork之后的子进程,在调用exec或exit之前,在父进程的空间中执行。
执行顺序
vfork保证子进程优先执行到exec或exit之前,父进程都不会被调度;fork父子进程执行顺序不确定。
因此,执行了vfork之后,子进程请立即执行exec,而不要再执行一次fork,否则就可能导致死锁。
或者这么说,如果在exec或exit之前依赖于父进程的进一步动作,就会导致死锁。
另外请留意,exec并不是创建进程,只是用新程序替换了当前进程的上下文。

int system(const char * cmdstring)
{
    pid_t pid;
    int status;
 
    if(cmdstring == NULL)
    {
        return (1); //如果cmdstring为空,返回非零值,一般为1
    }
    if((pid = fork())<0)
    {
        status = -1; //fork失败,返回-1
    }
    else if(pid == 0)
    {
        execl("/bin/sh", "sh", "-c", cmdstring, (char *)0);
        _exit(127); // exec执行失败返回127,注意exec只在失败时才返回现在的进程,成功的话现在的进程就不存在了
    }
    else //父进程
    {
        while(waitpid(pid, &status, 0) < 0)
        {
            if(errno != EINTR)
            {
                status = -1; //如果waitpid被信号中断,则返回-1
                break;
            }
        }
    }
     
    return status; //如果waitpid成功,则返回子进程的返回状态
}

参考自# :https://segmentfault.com/a/1190000005900042

 

  • 0
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
OS error code 0: Success 操作系统错误代码0:成功 OS error code 1: Operation not permitted 操作系统错误代码1:操作不允许 OS error code 2: No such file or directory 操作系统错误代码2:没有这样的文件或目录 OS error code 3: No such process 操作系统错误代码3:没有这样的过程 OS error code 4: Interrupted system call 操作系统错误代码4:中断的系统调用 OS error code 5: Input/output error 操作系统错误代码5:输入/输出错误 OS error code 6: No such device or address 操作系统错误代码6:没有这样的设备或地址 OS error code 7: Argument list too long 操作系统错误代码7:参数列表太长 OS error code 8: Exec format error 操作系统错误代码8:执行格式错误 OS error code 9: Bad file descriptor 操作系统错误代码9:坏的文件描述符 OS error code 10: No child processes 操作系统错误代码10:无子过程 OS error code 11: Resource temporarily unavailable 操作系统错误代码11:资源暂时不可用 OS error code 12: Cannot allocate memory 操作系统错误代码12:无法分配内存 OS error code 13: Permission denied 操作系统错误代码13:权限被拒绝 OS error code 14: Bad address 操作系统错误代码14:错误的地址 OS error code 15: Block device required 操作系统错误代码15:需要块设备 OS error code 16: Device or resource busy 操作系统错误代码16:设备或资源忙 OS error code 17: File exists 操作系统错误代码17:文件已经存在 OS error code 18: Invalid cross-device link 操作系统错误代码18:无效的跨设备链接 OS error code 19: No such device 操作系统错误代码19:没有这样的设备 OS error code 20: Not a directory 操作系统错误代码20:不是一个目录 OS error code 21: Is a directory 操作系统错误代码21:是一个目录 OS error code 22: Invalid argument 操作系统错误代码22:无效参数 OS error code 23: Too many open files in system 操作系统错误代码23:打开太多的文件系统 OS error code 24: Too many open files 操作系统错误代码24:打开的文件太多 OS error code 25: Inappropriate ioctl for device

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值