Redis申请内存出错:排查与解决
Redis是一个常用的内存数据库,应用广泛。然而,在Redis使用时,往往会遇到程序申请内存出错的情况。这种情况可能会影响Redis的性能,甚至导致Redis崩溃。本文将从排查错误的角度,介绍Redis申请内存出错的原因和解决方法。
错误信息:
在使用Redis时,若出现内存申请错误,Redis会在控制台输出类似如下的错误信息:
OOOMMAND GET key1 OOM command not allowed when used memory > ‘maxmemory’.
Cron may have encountered a permissions issue when it tried to create a temporary file. Verify that the cron user has permissions to create files in the specified directory.
这种错误信息往往会出现在Redis内存占用已经接近或超过所设置的最大内存限制时。Redis对内存的操作是使用C语言的malloc和free函数实现的。在程序执行过程中,如果Redis尝试申请超出可用内存的空间,则会出现内存申请错误。
原因分析:
Redis申请内存过程中出现错误的原因可能有很多。下面列举了几个常见的情况:
1. Redis的最大内存限制设置不当。如果限制设置太小,则会导致Redis内存占用过高,进而出现内存申请错误。
2. Redis的配置文件中,maxmemory-policy参数设置不当。maxmemory-policy参数是Redis用来控制内存回收的策略。如果该参数设置不当,则会出现内存不足的情况。
3. Redis频繁执行字符串拼接等操作。字符串的拼接操作可能会导致内存占用过高,进而出现内存申请错误。在实际开发中,应尽量避免这种操作。
解决方法:
针对上述情况,可以采取如下措施:
1. 修改maxmemory参数。在Redis配置文件redis.conf中,可以直接修改maxmemory参数的值。一般情况下,需要根据实际情况调整该值,避免出现内存申请错误。
2. 修改maxmemory-policy参数。如果Redis运行时,频繁出现内存不足的情况,可以尝试修改maxmemory-policy参数。该参数有多个取值:noeviction、allkeys-lru、volatile-lru、allkeys-random、volatile-random和volatile-ttl。其中,noeviction表示不回收内存;allkeys-lru表示回收所有数据中,最近最少使用的数据;volatile-lru表示回收过期数据中,最近最少使用的数据;allkeys-random和volatile-random表示回收数据中的任意键;volatile-ttl表示回收过期数据中,剩余时间最短的数据。一般情况下,建议选择volatile-lru。
3. 避免频繁执行字符串拼接等操作。在Redis开发过程中,应尽量避免使用字符串拼接等操作,以减少内存占用。如果确实需要进行字符串拼接,可以使用Redis提供的命令append,避免频繁执行字符串拼接操作。
代码示例:
下面是在Redis中使用append命令进行字符串拼接的示例代码:
> set key1 helloOK> append key1, world!OK> get key1"hello, world!"
在代码中,使用set命令将字符串hello写入Redis中的key1键,然后使用append命令将字符串world!拼接到key1键中,最后使用get命令获取key1键的值,可以看到拼接结果为hello, world!。
总结:
本文介绍了Redis申请内存出错的原因和解决方法。在Redis开发过程中,应该认真调整配置参数,避免使用字符串拼接等耗费大量内存的操作。同时,我们也可以使用Redis提供的append命令实现字符串拼接。只有这样,才能保障Redis的正常运行和性能。
香港服务器首选树叶云,2H2G首月10元开通。树叶云(shuyeidc.com)提供简单好用,价格厚道的香港/美国云 服务器 和独立服务器。IDC+ISP+ICP资质。ARIN和APNIC会员。成熟技术团队15年行业经验。
在编程中,“错误”和“非法”有什么区别?
编程中的非法情况通常是编程人员疏忽而导致的,是可避免的. 而错误是代码执行的结果与预期的不一致. 是无法避免的, 所以要做相应的处理. 例如:1) char *pszBuf = new char(100);2) memset(pszBuf, 0, 100);第一句话是申请内存操作,有两种结果:成功和失败.第二句是初始化申请到的内存. 如果申请内存失败就产生了错误, 如果不对这个错误进行处理就会产生非法情况(即异常). 所以代码应该这样写:char *pszBuf = new char(100);if (!pszBuf){return FALSE;}memset(pszBuf, 0, 100);这样就避免非法的产生啦!
同时,非法是通不过编译的,往往在编译阶段就报错了,
而错误有可能在运行的时候出现
电脑应用程序错误
0x????????”指令引用的“0x????????”内存。该内存不能为“read”。
“0x????????”指令引用的“0x????????”内存,该内存不能为“written”。
以上的情况相信大家都应该见到过,甚至说一些网友因为不爽于这个经常出现的错误提示而屡次重装系统。相信普通用户应该不会理解那些复杂的十六进制代码。
出现这个现象有方面的,一是硬件,即内存方面有问题,二是软件,这就有多方面的问题了。
一.先说说硬件:
一般来说,电脑硬件是很不容易坏的。内存出现问题的可能性并不大(除非你的内存真的是杂牌的一塌徒地),主要方面是:
1。内存条坏了(二手内存情况居多)
2。使用了有质量问题的内存。
3。内存插在主板上的金手指部分灰尘太多。
4。使用不同品牌不同容量的内存,从而出现不兼容的情况。
5。超频带来的散热问题。你可以使用MemTest 这个软件来检测一下内存,它可以彻底的检测出内存的稳定度。
二、如果都没有,那就从软件方面排除故障了。
先说原理:内存有个存放数据的地方叫缓冲区,当程序把数据放在缓冲区,需要操作系统提供的“功能函数”来申请,如果内存分配成功,函数就会将所新开辟的内存区地址返回给应用程序,应用程序就可以通过这个地址使用这块内存。这就是“动态内存分配”,内存地址也就是编程中的“光标”。内存不是永远都招之即来、用之不尽的,有时候内存分配也会失败。当分配失败时系统函数会返回一个0值,这时返回值“0”已不表示新启用的光标,而是系统向应用程序发出的一个通知,告知出现了错误。作为应用程序,在每一次申请内存后都应该检查返回值是否为0,如果是,则意味着出现了故障,应该采取一些措施挽救,这就增强了程序的“健壮性”。若应用程序没有检查这个错误,它就会按照“思维惯性”认为这个值是给它分配的可用光标,继续在之后的执行中使用这块内存。真正的0地址内存区储存的是计算机系统中最重要的“中断描述符表”,绝对不允许应用程序使用。在没有保护机制的操作系统下(如DOS),写数据到这个地址会导致立即当机,而在健壮的操作系统中,如Windows等,这个操作会马上被系统的保护机制捕获,其结果就是由操作系统强行关闭出错的应用程序,以防止其错误扩大。这时候,就会出现上述的内存不能为“read”错误,并指出被引用的内存地址为“0x“。内存分配失败故障的原因很多,内存不够、系统函数的版本不匹配等都可能有影响。因此,这种分配失败多见于操作系统使用很长时间后,安装了多种应用程序(包括无意中“安装”的病毒程序),更改了大量的系统参数和系统档案之后。
在使用动态分配的应用程序中,有时会有这样的情况出现:程序试图读写一块“应该可用”的内存,但不知为什么,这个预料中可用的光标已经失效了。有可能是“忘记了”向操作系统要求分配,也可能是程序自己在某个时候已经注销了这块内存而“没有留意”等等。注销了的内存被系统回收,其访问权已经不属于该应用程序,因此读写操作也同样会触发系统的保护机制,企图“违法”的程序唯一的下场就是被操作终止执行,回收全部资源。计算机世界的法律还是要比人类有效和严厉得多啊!像这样的情况都属于程序自身的BUG,你往往可在特定的操作顺序下重现错误。无效光标不一定总是0,因此错误提示中的内存地址也不一定为“0x”,而是其它随机数字。
首先建议:
1、 检查系统中是否有木马或病毒。这类程序为了控制系统往往不负责任地修改系统,从而导致操作系统异常。平常应加强信息安全意识,对来源不明的可执行程序绝不好奇。
2、 更新操作系统,让操作系统的安装程序重新拷贝正确版本的系统档案、修正系统参数。有时候操作系统本身也会有BUG,要注意安装官方发行的升级程序。
3、 尽量使用最新正式版本的应用程序、Beta版、试用版都会有BUG。
4、 删除然后重新创建 Winnt\System32\Wbem\Repository 文件夹中的文件:在桌面上右击我的电脑,然后单击管理。 在服务和应用程序下,单击服务,然后关闭并停止 Windows Management Instrumentation 服务。 删除 Winnt\System32\Wbem\Repository 文件夹中的所有文件。(在删除前请创建这些文件的备份副本。) 打开服务和应用程序,单击服务,然后打开并启动 Windows Management Instrumentation 服务。当服务重新启动时,将基于以下注册表项中所提供的信息重新创建这些文件: HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\WBEM\CIMOM\Autorecover MOFs

winlogon.exe-应用程序出错 (不懂行的别进)
出现这个现象有两方面的,一是硬件,即内存方面有问题,二是软件,这就有多方面的问题了。 一:先说说硬件: 一般来说,电脑硬件是很不容易坏的。 内存出现问题的可能性并不大(除非你的内存真的是杂牌的一塌徒地),主要方面是:1。 内存条坏了(二手内存情况居多)、2。 使用了有质量问题的内存,3。 内存插在主板上的金手指部分灰尘太多。 4。 使用不同品牌不同容量的内存,从而出现不兼容的情况。 5。 超频带来的散热问题。 你可以使用MemTest 这个软件来检测一下内存,它可以彻底的检测出内存的稳定度。 二、如果都没有,那就从软件方面排除故障了。 先说原理:内存有个存放数据的地方叫缓冲区,当程序把数据放在缓冲区,需要操作系统提供的“功能函数”来申请,如果内存分配成功,函数就会将所新开辟的内存区地址返回给应用程序,应用程序就可以通过这个地址使用这块内存。 这就是“动态内存分配”,内存地址也就是编程中的“光标”。 内存不是永远都招之即来、用之不尽的,有时候内存分配也会失败。 当分配失败时系统函数会返回一个0值,这时返回值“0”已不表示新启用的光标,而是系统向应用程序发出的一个通知,告知出现了错误。 作为应用程序,在每一次申请内存后都应该检查返回值是否为0,如果是,则意味着出现了故障,应该采取一些措施挽救,这就增强了程序的“健壮性”。 若应用程序没有检查这个错误,它就会按照“思维惯性”认为这个值是给它分配的可用光标,继续在之后的执行中使用这块内存。 真正的0地址内存区储存的是计算机系统中最重要的“中断描述符表”,绝对不允许应用程序使用。 在没有保护机制的操作系统下(如DOS),写数据到这个地址会导致立即当机,而在健壮的操作系统中,如Windows等,这个操作会马上被系统的保护机制捕获,其结果就是由操作系统强行关闭出错的应用程序,以防止其错误扩大。 这时候,就会出现上述的内存不能为“read”错误,并指出被引用的内存地址为“0x“。 内存分配失败故障的原因很多,内存不够、系统函数的版本不匹配等都可能有影响。 因此,这种分配失败多见于操作系统使用很长时间后,安装了多种应用程序(包括无意中“安装”的病毒程序),更改了大量的系统参数和系统档案之后。 在使用动态分配的应用程序中,有时会有这样的情况出现:程序试图读写一块“应该可用”的内存,但不知为什么,这个预料中可用的光标已经失效了。 有可能是“忘记了”向操作系统要求分配,也可能是程序自己在某个时候已经注销了这块内存而“没有留意”等等。 注销了的内存被系统回收,其访问权已经不属于该应用程序,因此读写操作也同样会触发系统的保护机制,企图“违法”的程序唯一的下场就是被操作终止执行,回收全部资源。 计算机世界的法律还是要比人类有效和严厉得多啊!像这样的情况都属于程序自身的BUG,你往往可在特定的操作顺序下重现错误。 无效光标不一定总是0,因此错误提示中的内存地址也不一定为“0x”,而是其它随机数字。 首先建议: 1、 检查系统中是否有木马或病毒。 这类程序为了控制系统往往不负责任地修改系统,从而导致操作系统异常。 平常应加强信息安全意识,对来源不明的可执行程序绝不好奇。 2、 更新操作系统,让操作系统的安装程序重新拷贝正确版本的系统档案、修正系统参数。 有时候操作系统本身也会有BUG,要注意安装官方发行的升级程序。 3、 尽量使用最新正式版本的应用程序、Beta版、试用版都会有BUG。 4、 删除然后重新创建 Winnt\System32\Wbem\Repository 文件夹中的文件:在桌面上右击我的电脑,然后单击管理。 在服务和应用程序下,单击服务,然后关闭并停止 Windows Management Instrumentation 服务。 删除 Winnt\System32\Wbem\Repository 文件夹中的所有文件。 (在删除前请创建这些文件的备份副本。 ) 打开服务和应用程序,单击服务,然后打开并启动 Windows Management Instrumentation 服务。 当服务重新启动时,将基于以下注册表项中所提供的信息重新创建这些文件: HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\WBEM\CIMOM\Autorecover MOFs
发表评论