V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
sharkv
V2EX  ›  Redis

新手关于 redis 丢数据的问题

  •  
  •   sharkv · 2019-03-14 15:02:32 +08:00 · 10132 次点击
    这是一个创建于 2074 天前的主题,其中的信息可能已经有所发展或是发生改变。

    问题是这样的: 本人在 springboot 项目中加入了 redis,并且能够存取 redis 中的数据。 我设置了 300s 的有效时间,我一直用 redis-client 刷新数据的有效时间, 但是它时间还没到数据就消失了,报的错误 key not exist: [email protected]

    application.yml 中配置了 host、port、password 代码是这样的

    import org.springframework.cache.CacheManager;
    import org.springframework.cache.annotation.EnableCaching;
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.data.redis.cache.RedisCacheManager;
    import org.springframework.data.redis.connection.RedisConnectionFactory;
    import org.springframework.data.redis.core.RedisTemplate;
    import org.springframework.data.redis.serializer.RedisSerializer;
    import org.springframework.data.redis.serializer.StringRedisSerializer;
    
    @Configuration
    @EnableCaching
    public class RedisCacheConfig {
        @Bean
        public CacheManager cacheManager(RedisTemplate<?, ?> redisTemplate){
            CacheManager cacheManager = new RedisCacheManager(redisTemplate);
            return cacheManager;
        }
        @Bean
        public RedisTemplate<String, String> redisTemplate(RedisConnectionFactory factory){
            RedisTemplate<String, String> redisTemplate = new RedisTemplate<String,String>();
            redisTemplate.setConnectionFactory(factory);
            // key 序列化方式;(不然会出现乱码;),但是如果方法上有 Long 等非 String 类型的话,会报类型转换错误;
            // 所以在没有自己定义 key 生成策略的时候,以下这个代码建议不要这么写,可以不配置或者自己实现 ObjectRedisSerializer
            // 或者 JdkSerializationRedisSerializer 序列化方式;
            RedisSerializer<String> redisSerializer = new StringRedisSerializer();// Long 类型不可以会出现异常信息;
            redisTemplate.setKeySerializer(redisSerializer);
            redisTemplate.setHashKeySerializer(redisSerializer);
            return redisTemplate;
        }
    
    }
    
    

    存入数据:

    redisTemplate.opsForValue().set(express, captcha, 300, TimeUnit.SECONDS);
    

    取出数据:

    localCaptcha = redisTemplate.opsForValue().get(key);
    
    10 条回复    2019-03-14 20:43:25 +08:00
    yc8332
        1
    yc8332  
       2019-03-14 15:31:53 +08:00
    不是可以通过 ttl 看设置的过期时间吗? ttl key 看设置的过期时间
    Raymon111111
        2
    Raymon111111  
       2019-03-14 15:32:37 +08:00
    先上服务端看看这个 key 是不是存在
    Raymon111111
        3
    Raymon111111  
       2019-03-14 15:35:08 +08:00   ❤️ 2
    我解决的话会先定位问题点

    1. set 这个方法的有效性. 调用了 set 之后服务端能否看见这个, 如果不可以, 先解决 set 写的不对的问题.

    2. 如果 set 没问题, 那么再看 set 上过期时间是不是正确的, set 完之后上服务端看看过期时间是不是能对上.

    3. 如果 set 整个没问题, 就检查 get 方法的毛病...

    ...反正分步排查吧
    boris1993
        4
    boris1993  
       2019-03-14 15:41:36 +08:00 via Android
    同意#3,调查写没写进去,超时有没有设置上,超时时间的单位是什么,确定写操作都正确再检查为啥读不到
    sharkv
        5
    sharkv  
    OP
       2019-03-14 16:08:09 +08:00
    这个不好贴图。我都试过了,能在 redis 可视化客户端看到数据存在过,然后刷新一段时间就没了。我找到的原因是 redis 里面会自动存入 caches 和 runtime 两条数据,这两个数据一存在我的数据就会丢失。
    sharkv
        6
    sharkv  
    OP
       2019-03-14 16:09:08 +08:00
    我在 redis 把 caches 和 runtime 删了过段时间它又自动生成了。所以该怎么删。。
    sharkv
        7
    sharkv  
    OP
       2019-03-14 16:28:37 +08:00
    问题找到了。应该就是被人撞库了,我改了下密码,设置了下 ip 就可以了
    boris1993
        8
    boris1993  
       2019-03-14 16:31:59 +08:00 via Android
    @sharkv #7 你 Redis......监听公网还不上密码???
    cheava
        9
    cheava  
       2019-03-14 19:58:52 +08:00
    @sharkv #7 这问题还真不在一般的 debug 范围内.....大家都默认了你的 redis 和应用在同一内网......
    Leigg
        10
    Leigg  
       2019-03-14 20:43:25 +08:00 via iPhone
    写个脚本,每隔一秒读取一下这个值,set 后马上执行脚本。
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   1181 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 30ms · UTC 18:42 · PVG 02:42 · LAX 10:42 · JFK 13:42
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.