Spring Boot 集成 Redis 并调用 Lua 脚本详解

发布于:2025-07-17 ⋅ 阅读:(10) ⋅ 点赞:(0)

前言

Redis 是一个高性能的内存数据库,广泛用于缓存、计数器、分布式锁等场景。在某些业务中,我们需要对多个 Redis 操作进行原子性处理,以保证数据一致性,这时就可以使用 Redis 提供的 Lua 脚本功能。

Spring Boot 通过 RedisTemplate 对 Redis 进行了良好的封装,也支持我们方便地调用 Lua 脚本。本文将带你一步步实现 Spring Boot 中如何:

  • 加载 Lua 脚本
  • 调用 Lua 脚本
  • 实现一个实际业务逻辑:获取 key,若不存在则设置默认值并返回

一、项目准备

1. 创建 Spring Boot 项目

你可以使用 Spring Initializr 创建一个新项目,选择以下依赖:

  • Spring Web
  • Spring Data Redis
  • Lettuce(Redis 客户端)

或者手动创建 Maven 项目后添加如下依赖:

<!-- Spring Boot Starter -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter</artifactId>
</dependency>

<!-- Web 支持 -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>

<!-- Redis 支持 -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

<!-- Lettuce 客户端 -->
<dependency>
    <groupId>io.lettuce.core</groupId>
    <artifactId>lettuce-core</artifactId>
    <version>6.2.4</version>
</dependency>

二、编写 Lua 脚本文件

我们将创建一个简单的 Lua 脚本:如果 key 不存在,则设置默认值;存在则直接返回当前值。

脚本内容:get_or_set_default.lua

路径建议为:src/main/resources/lua/get_or_set_default.lua

-- get_or_set_default.lua
local key = KEYS[1]
local defaultVal = ARGV[1]

local value = redis.call('GET', key)
if not value then
    redis.call('SET', key, defaultVal)
    value = defaultVal
end
return value

三、加载 Lua 脚本到 Redis

我们可以利用 Spring 的生命周期回调(如 @PostConstruct)来加载 Lua 脚本,并保存其 SHA1 校验码,后续通过 EVALSHA 来调用。

创建脚本加载类:LuaScriptLoader.java

import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.stereotype.Service;

import javax.annotation.PostConstruct;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Collections;

@Service
public class LuaScriptLoader {

    private final RedisTemplate<String, Object> redisTemplate;

    // 存储脚本的 SHA1 值
    public static String GET_OR_SET_DEFAULT_SHA;

    public LuaScriptLoader(RedisTemplate<String, Object> redisTemplate) {
        this.redisTemplate = redisTemplate;
    }

    @PostConstruct
    public void loadScripts() throws IOException {
        Path scriptPath = Paths.get("src/main/resources/lua/get_or_set_default.lua");
        String scriptContent = Files.readString(scriptPath);

        DefaultRedisScript<String> redisScript = new DefaultRedisScript<>();
        redisScript.setScriptText(scriptContent);
        redisScript.setResultType(String.class);

        // 执行 SCRIPT LOAD 命令加载脚本
        GET_OR_SET_DEFAULT_SHA = redisTemplate.execute(
                redisScript,
                Collections.singletonList("dummyKey"),
                "defaultValue"
        );

        System.out.println("Lua Script SHA1: " + GET_OR_SET_DEFAULT_SHA);
    }
}

注意:

  • 我们使用了一个 dummyKey 和 dummyValue 来执行一次脚本,只是为了触发脚本加载。
  • 实际调用时使用的是 EVALSHA 命令。

四、封装 Lua 脚本调用服务

我们创建一个服务类来封装 Lua 脚本的调用逻辑。

创建服务类:RedisLuaService.java

import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;

import java.util.Collections;

@Service
public class RedisLuaService {

    private final RedisTemplate<String, Object> redisTemplate;

    public RedisLuaService(RedisTemplate<String, Object> redisTemplate) {
        this.redisTemplate = redisTemplate;
    }

    /**
     * 调用 Lua 脚本:根据 key 获取值,若不存在则设置默认值并返回
     */
    public String getOrSetDefault(String key, String defaultValue) {
        return (String) redisTemplate.execute(
                LuaScriptLoader.GET_OR_SET_DEFAULT_SHA,
                Collections.singletonList(key),
                defaultValue
        );
    }
}

五、创建 REST 接口进行测试

为了方便测试,我们创建一个简单的接口。

创建控制器类:RedisLuaController.java

import org.springframework.web.bind.annotation.*;

@RestController
@RequestMapping("/redis")
public class RedisLuaController {

    private final RedisLuaService redisLuaService;

    public RedisLuaController(RedisLuaService redisLuaService) {
        this.redisLuaService = redisLuaService;
    }

    @GetMapping("/get-or-set")
    public String getOrDefault(
            @RequestParam String key,
            @RequestParam(required = false, defaultValue = "default_value") String defaultValue) {
        return redisLuaService.getOrSetDefault(key, defaultValue);
    }
}

六、运行测试

1. 启动 Redis 服务

确保你本地或服务器上已经启动了 Redis:

redis-server

2. 启动 Spring Boot 应用

使用 IDE 或命令行启动 Spring Boot 项目:

mvn spring-boot:run

3. 测试访问

访问如下 URL:

http://localhost:8080/redis/get-or-set?key=test_lua&defaultValue=hello_redis
第一次请求:
  • Redis 中没有 test_lua 键,会自动设置为 hello_redis,并返回该值。
第二次请求:
  • 即使不传 defaultValue,也会返回 hello_redis,因为键已存在。

七、注意事项

项目 说明
脚本缓存 Redis 会缓存加载过的 Lua 脚本,但重启后失效,需重新加载
调试困难 Lua 脚本不能输出日志,建议先在 Redis CLI 中测试后再集成
性能考量 Lua 脚本是单线程执行的,避免复杂运算
异常处理 在 Java 层应对脚本执行失败做容错处理
参数传递 KEYS 是数组,ARGV 是参数数组,顺序要对应

八、进阶建议

  • 将所有 Lua 脚本统一管理,使用配置中心或数据库存储
  • 结合 AOP 或拦截器,实现 Lua 脚本调用的自动重试机制
  • 使用 Redisson 等高级库简化 Lua 脚本操作
  • 使用 Spring Cache 抽象层结合 Lua 脚本实现更复杂的缓存策略

示例结构图

src/
├── main/
│   ├── java/
│   │   └── com.example.demo/
│   │       ├── DemoApplication.java
│   │       ├── service/
│   │       │   ├── RedisLuaService.java
│   │       │   └── LuaScriptLoader.java
│   │       └── controller/
│   │           └── RedisLuaController.java
│   └── resources/
│       └── lua/
│           └── get_or_set_default.lua
└── pom.xml

总结

本文详细介绍了如何在 Spring Boot 中调用 Redis 的 Lua 脚本,包括:

  • 添加必要的依赖
  • 编写 Lua 脚本
  • 使用 RedisTemplate 加载和调用脚本
  • 构建完整的调用流程与接口测试

Lua 脚本非常适合用来实现一些需要原子性的 Redis 操作,是构建高并发系统的重要工具之一。


网站公告

今日签到

点亮在社区的每一天
去签到