redis解决Hincrbyfloat精度丢失问题

在使用redis缓存金额相关数据时,为了保证操作的原子性,通常会使用 hincrByFloat 来操作,单是 hincrByFloat 有时会存在精度丢失问题,可以用过lua脚本来处理,即可以保证原子性也能保证精度不丢失
在这里插入图片描述

@Test
    public void lua() {
    
    
        String key = "api_sdk_entity:quota_cont:ping001";
        String filed ="sumQuota";
        BigDecimal price = new BigDecimal("100.01");
        List<String> keys = new ArrayList<String>();
        keys.add(key);
        List<String> args = new ArrayList<String>();
        args.add(filed);
        args.add(price.toPlainString());
        System.out.println("======1======"+jcRedisCache.hget(key,filed));
        //--创建val变量并查询hash的数据
        String script ="local val = redis.call('hGet',KEYS[1],ARGV[1]);" +
                //--创建inc_val变量此值就是自增后的值
                "local inc_val;" +
                //--判断val是否存在,存在则增加值,不存在则直接是传过来的值,此处要注意val的redis返回值如果不存在key返回的是nil但是转成lua的结果是 false
                "if val ~=false then " +
                    "inc_val =  tostring(val + ARGV[2]) " +
                "else " +
                    "inc_val =  tostring(ARGV[2])  " +
                "end;" +
                //--操作redis重新赋值
                "local res =redis.call('hSet',KEYS[1],ARGV[1],inc_val);" +
                "if res >= 0 then " +
                    "return inc_val " +
                "else " +
                    "return res " +
                "end;";
        jcRedisCache.commond(script,keys,args);
        System.out.println("======2======"+jcRedisCache.hget(key,filed));
    }
======
 
  @Override
    public String commond(String script,List<String> keys,List<String> args) {
    
    
        String sha = client.scriptLoad(script);
        client.evalsha(sha,keys, args);
        return null;
    }

猜你喜欢

转载自blog.csdn.net/wenwang3000/article/details/127159061