Springboot redis luttuce lua脚本 存储压缩的protobuf key:byte[] value:map<byte,byte[]>

注释部分是正常的map<string,value> 的方式,对该lua脚本进行了优化

复制代码
 /**
     *  lua 脚本 实现物料key 的赋值
     * @param hashKey 物料key
     * @param fieldsAndValues lua 脚本参数
     * @return
     */
    public Long insertIntoHashWithExpireTime(byte[] hashKey, Map<byte[], byte[]> fieldsAndValues, int expireTime) {
        Long result = null;
        // unpack(ARGV)在Lua中是用来将数组解包成一系列单独的参数。
        // ARGV[#ARGV]获取的是传递给脚本的最后一个参数(在 Lua 中,# 操作符用于获取表中的元素数量。因此,ARGV[#ARGV] 将返回 ARGV 表中的最后一个元素。)
        // 请求案例:insertIntoHashWithExpireTimeIfKeyExists("myHash", "field1", "value1", "field2", "value2", 60);
//        String luaScript =
//                "\nlocal argv = ARGV \n" +
//                        "local length = #argv \n"+
//                        // 将1 - n-1 的入参写入新的数组
//                        "if length > 0 then  local unpackArgs = {} \n"  +
//                        "for i = 1, length - 1 do  \n" +
//                        " table.insert(unpackArgs, argv[i]) \n" +
//                        "end\n" +
//                        "if redis.call('exists', KEYS[1]) == 1 then \n" +
//                        "\tredis.call('del', KEYS[1]) \n" +
//                        "\tredis.call('hset', KEYS[1], unpack(unpackArgs)) \n" +
//                        "\tredis.call('expire', KEYS[1], argv[length]) \n" +
//                        "\treturn 1  \n" +
//                        "else \n" +
//                        "\tredis.call('hset', KEYS[1], unpack(unpackArgs)) \n" +
//                        "\tredis.call('expire', KEYS[1], argv[length]) \n" +
//                        "\treturn 2\n" +
//                        "end \n" +
//                        "end ";
        String luaScript =
                "local argv = ARGV\n" +
                        "local length = #argv\n" +
                        "\n" +
                        "-- 检查是否有参数传入\n" +
                        "if length > 0 then\n" +
                        "    -- 创建一个新的表来存储键值对\n" +
                        "    local unpackArgs = {}\n" +
                        "\n" +
                        "    -- 从第一个参数开始处理\n" +
                        "    -- 将ARGV中的键值对转换为Lua表\n" +
                        "    for i = 1, length - 1, 2 do\n" +
                        "        table.insert(unpackArgs, argv[i])     -- 键\n" +
                        "        table.insert(unpackArgs, argv[i + 1]) -- 值\n" +
                        "    end\n" +
                        "\n" +
                        "    -- 获取过期时间\n" +
                        "    local expireTime = tonumber(argv[length])\n" +
                        "\n" +
                        "    -- 检查键是否存在\n" +
                        "    if redis.call('exists', KEYS[1]) == 1 then\n" +
                        "        -- 如果键存在,先删除旧的键\n" +
                        "        redis.call('del', KEYS[1])\n" +
                        "        -- 再设置新的键值对\n" +
                        "        for i = 1, #unpackArgs - 1, 2 do\n" +
                        "            redis.call('hset', KEYS[1], unpackArgs[i], unpackArgs[i + 1])\n" +
                        "        end\n" +
                        "        -- 设置过期时间\n" +
                        "        redis.call('expire', KEYS[1], expireTime)\n" +
                        "        return 1\n" +
                        "    else\n" +
                        "        -- 如果键不存在,直接设置新的键值对\n" +
                        "        for i = 1, #unpackArgs - 1, 2 do\n" +
                        "            redis.call('hset', KEYS[1], unpackArgs[i], unpackArgs[i + 1])\n" +
                        "        end\n" +
                        "        -- 设置过期时间\n" +
                        "        redis.call('expire', KEYS[1], expireTime)\n" +
                        "        return 2\n" +
                        "    end\n" +
                        "end";
        log.info("luaScript:{}", luaScript);
        try {
            // 将 Map 转换成有序的键值对列表
            // 创建参数列表
            List<byte[]> args = new ArrayList<>(fieldsAndValues.size() * 2 + 1);
            for (Map.Entry<byte[], byte[]> entry : fieldsAndValues.entrySet()) {
                args.add(entry.getKey());
                args.add(entry.getValue());
            }
            args.add(String.valueOf(expireTime).getBytes()); // 最后一个参数是过期时间

            DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>();
            redisScript.setResultType(Long.class);//返回类型是Long
            redisScript.setScriptText(luaScript);

            // 执行脚本
            result = protobufRedisTemplate.execute(redisScript,
                    Collections.singletonList(hashKey), // KEYS
                    args.toArray(new byte[0][])); // ARGV

            log.debug("redisEVALLuaScript result :{}", result);
        } catch (Exception e) {
            e.printStackTrace();
            log.error("e:", e);
        }
        return result;
    }

有一说一,感谢大模型,这个完全是大模型给出的答案。

结果如下

相关推荐
IT_Octopus7 小时前
Spring Boot 实战:@PostConstruct + Caffeine 缓存初始化与定时刷新
spring boot·后端·缓存
yangshicong8 小时前
第11章:结构化输出与数据提取 —— 让 AI 直接返回你想要的数据格式
数据库·人工智能·redis·python·langchain·ai编程
一路向北·重庆分伦9 小时前
09:Redis-高级特性与原理
redis
zb2006412011 小时前
Laravel7.x十大核心特性解析
spring boot·后端·laravel
他们叫我阿冠11 小时前
Day5学习--SpringBoot详解
spring boot·后端·学习
qq_2651533711 小时前
Redis在游戏服务器中怎么实现开合服数据同步?
服务器·redis·游戏·游戏服务器
vx-程序开发13 小时前
基于机器学习的动漫可视化系统的设计与实现-计算机毕业设计源码08339
java·c++·spring boot·python·spring·django·php
ElevenS_it18815 小时前
Redis监控实战:内存使用+命中率+连接数三类核心指标接入Zabbix+分级告警完整配置方案
运维·网络·redis·mybatis·zabbix
逍遥德16 小时前
Java编程高频的“技术点”-01:自定义全局异常处理器
java·开发语言·spring boot·后端
小旭952717 小时前
商品详情实现与缓存问题(穿透、击穿、雪崩)解决方案
java·数据库·spring boot·后端·缓存