redis设置过期时间是为数据添加自动清理机制以节省内存并提升性能。1.使用expire命令可灵活设置秒级过期时间;2.pexpire命令用于毫秒级精度控制;3.setex命令可在设置值的同时指定过期时间,一步到位;4.expireat命令按指定时间戳过期。此外,ttl和pttl命令可用于查看剩余生存时间,redis采用惰性删除与定期删除结合的策略清理过期键。设置时需注意合理选择过期时长、避免集中过期、考虑数据类型影响及持久化配置,批量设置可通过lua脚本实现。
redis设置过期时间,简单来说,就是给你的数据加上一个“保质期”。到期了,redis会自动把它清理掉,节省内存空间,保持数据库的活力。
Redis提供了多种方式来设置过期时间,下面就来详细聊聊这几种方法,以及一些需要注意的地方。
为什么需要设置过期时间?
想象一下,如果你的Redis里塞满了永远不会被删除的数据,那会发生什么?内存迟早会被耗尽,Redis的性能也会直线下降。设置过期时间,就像给你的数据添加了一个自动清理机制,可以有效避免这种情况。比如,你可以用它来存储用户的登录Token,设置一个小时的有效期,过期后自动失效,无需手动清理。或者缓存一些临时数据,比如某个API的调用结果,设置一个较短的过期时间,减轻数据库的压力。
4种设置过期时间的实用技巧
Redis提供了多种设置过期时间的方式,每种方式都有其适用的场景。
-
EXPIRE命令:简单直接,灵活设置
这是最常用的方法,直接在键上设置过期时间,单位是秒。
SET mykey "hello" EXPIRE mykey 60 # 设置mykey在60秒后过期
这种方式非常灵活,可以随时给已存在的键设置或修改过期时间。但是,要注意的是,如果键被修改(比如使用了SET命令),那么它的过期时间会被移除。
-
PEXPIRE命令:毫秒级的精度控制
PEXPIRE命令和EXPIRE命令类似,只不过它的单位是毫秒。如果你需要更精确的控制过期时间,比如需要设置一个3.5秒的过期时间,那么PEXPIRE就派上用场了。
SET mykey "hello" PEXPIRE mykey 3500 # 设置mykey在3.5秒后过期
-
SETEX命令:一步到位,设置值和过期时间
SETEX命令可以一次性完成设置键值和过期时间的操作,避免了先SET再EXPIRE的两步操作,更简洁高效。
SETEX mykey 60 "hello" # 设置mykey的值为"hello",并在60秒后过期
需要注意的是,SETEX命令只能设置秒级的过期时间。
-
EXPIREAT命令:指定过期的时间戳
EXPIREAT命令允许你指定键的过期时间为一个unix时间戳。这意味着你可以根据一个确定的时间点来设置过期时间,而不是一个相对的时间间隔。
SET mykey "hello" EXPIREAT mykey 1678886400 # 设置mykey在2023年3月16日0点0分0秒过期
这种方式在某些场景下非常有用,比如你需要批量设置一批键的过期时间,并且这些键都需要在同一时间过期。
如何查看键的剩余生存时间?
有时候,你可能需要查看某个键还剩下多少时间过期。可以使用TTL命令或者PTTL命令来查看。TTL命令返回剩余的秒数,PTTL命令返回剩余的毫秒数。如果键不存在,或者已经过期,TTL命令会返回-2,PTTL命令也会返回-2。如果键存在但没有设置过期时间,TTL命令会返回-1,PTTL命令也会返回-1。
SET mykey "hello" EXPIRE mykey 60 TTL mykey # 返回剩余的秒数,比如55 PTTL mykey # 返回剩余的毫秒数,比如55234
键的过期策略:Redis是如何清理过期键的?
Redis采用了一种惰性删除和定期删除相结合的过期策略。
-
惰性删除: 当你尝试访问一个已经过期的键时,Redis会先检查它是否过期,如果过期,则删除它,然后返回空。这种方式的优点是节省CPU资源,只在访问时才进行检查。缺点是如果一个键过期了,但是一直没有被访问,那么它会一直占用内存。
-
定期删除: Redis会定期(默认每秒10次)抽取一些键进行过期检查,删除过期的键。这种方式可以弥补惰性删除的不足,防止大量的过期键占用内存。但是,定期删除也会消耗一定的CPU资源。
设置过期时间需要注意什么?
- 选择合适的过期时间: 过期时间设置得太短,会导致缓存频繁失效,降低性能;设置得太长,又可能导致数据不一致。需要根据实际业务场景进行权衡。
- 避免大量键在同一时间过期: 如果大量的键在同一时间过期,可能会导致Redis瞬间压力过大,影响性能。可以考虑给不同的键设置不同的过期时间,避免集中过期。
- 注意键的数据类型: 不同的数据类型,过期时间的影响可能不同。比如,对于List、Set、Hash等数据类型,过期时间只针对整个键,而不是针对键中的某个元素。
- 持久化和过期时间: 如果Redis开启了持久化功能,那么过期时间也会被保存到持久化文件中。当Redis重启时,会根据持久化文件中的过期时间来恢复键的过期时间。
如何批量设置过期时间?
虽然Redis没有直接提供批量设置过期时间的命令,但你可以通过Lua脚本来实现批量设置过期时间的功能。Lua脚本可以在Redis服务器端执行,减少网络开销,提高效率。
-- keys: key的前缀 -- ARGV[1]: 过期时间,单位秒 local prefix = KEYS[1] local ttl = ARGV[1] local keys = redis.call('KEYS', prefix .. '*') for i, key in ipairs(keys) do redis.call('EXPIRE', key, ttl) end return keys
你可以使用
EVAL
命令来执行这个Lua脚本。
EVAL "local prefix = KEYS[1] local ttl = ARGV[1] local keys = redis.call('KEYS', prefix .. '*') for i, key in ipairs(keys) do redis.call('EXPIRE', key, ttl) end return keys" 1 myprefix 60
这个例子会给所有以
myprefix
开头的键设置60秒的过期时间。
总而言之,熟练掌握Redis的过期时间设置技巧,可以帮助你更好地管理Redis中的数据,提高Redis的性能和稳定性。根据不同的业务场景,选择合适的过期时间设置方式,并注意一些细节问题,才能充分发挥Redis的优势。