Spring Boot - Cache!
Spring Data redis
dependencies {
implementation 'org.springframework.boot:spring-boot-starter-data-redis'
}
application.properties
에 아래와 같이 설정 한 후 Bean 객체 생성이 필요하다.
redis.host=localhost
redis.port=6379
redis.timeout=0
@Slf4j
@Configuration
public class RedisConfig extends CachingConfigurerSupport {
@Value("${redis.host}")
private String host;
@Value("${redis.port}")
private int port;
@Bean
public RedisConnectionFactory redisConnectionFactory() {
return new LettuceConnectionFactory(host, port);
}
}
redis
서버와 연결을 위한 객체를 생성하는 RedisConnectionFactory
,
Spring Data Redis
프로젝트를 사용해서 CrudRepository
구현체를 통해
redis
에 저장할 도메인 객체를 정의할 수 있다.
아래와 같이 @RedisHash
어노테이션을 사용한다.
@AllArgsConstructor
@ToString
@RedisHash("point")
public class Point {
@Id
private String id;
private Long amount;
private LocalDateTime refreshTime;
public void refresh(Long amount, LocalDateTime refreshTime) {
if (refreshTime.isAfter(this.refreshTime)) {
this.amount = amount;
this.refreshTime = refreshTime;
}
}
}
public interface PointRedisRepository extends CrudRepository<Point, String> {
}
@Slf4j
@RestController
@RequestMapping("/repository")
@RequiredArgsConstructor
public class RedisRepositoryController {
private final PointRedisRepository repository;
@GetMapping("/{id}")
public Point getPointById(@PathVariable String id) {
return repository.findById(id)
.orElseThrow(() -> new IllegalArgumentException("invalid id:" + id));
}
@PostMapping
public Point addPoint(@RequestBody Point point) {
return repository.save(point);
}
@DeleteMapping("/{id}")
public void removePoint(@PathVariable String id) {
repository.deleteById(id);
}
@PatchMapping("/{id}")
public Point updatePoint(@PathVariable String id, @RequestBody Point point) {
Point entity = repository.findById(id)
.orElseThrow(() -> new IllegalArgumentException("invalid id:" + id));
entity.update(point);
entity= repository.save(entity);
return entity;
}
}
CrudRepository
통해 객체타입의 데이터를 redis
에 저장하면 hash
와 set
데이터구조로 관리한다.
아래와 같이 curl 로 2개의 point 를 저장하고 redis-cli
로 내부값을 살펴보면
2개의 hash 타입 데이터, 1개의 set 타입 데이터가 저장된 것을 확인할 수 있다.
curl --location 'http://127.0.0.1:8080/repository' \
--header 'Content-Type: application/json' \
--data '{
"id": "1",
"amount": "1000",
"refreshTime": "2023-03-22T11:11:11.000"
}'
curl --location 'http://127.0.0.1:8080/repository' \
--header 'Content-Type: application/json' \
--data '{
"id": "2",
"amount": "2000",
"refreshTime": "2023-03-22T22:22:22.000"
}'
$ redis-cli
127.0.0.1:6379> keys *
# 1) "point:1"
# 2) "point"
# 3) "point:2"
127.0.0.1:6379> smembers point
# 1) "1"
# 2) "2"
127.0.0.1:6379> HGETALL point:1
# 1) "_class"
# 2) "com.example.redis.model.Point"
# 3) "amount"
# 4) "1000"
# 5) "id"
# 6) "1"
# 7) "refreshTime"
# 8) "2023-03-22T11:11:11"
127.0.0.1:6379> HGETALL point:2
# 1) "_class"
# 2) "com.example.redis.model.Point"
# 3) "amount"
# 4) "2000"
# 5) "id"
# 6) "2"
# 7) "refreshTime"
# 8) "2023-03-22T22:22:22"
RedisTemplate
위에서 생성했던 RedisConnectionFactory
빈 객체를 주입해서 RedisTemplate
구성할 수 있다.
@Bean
public StringRedisTemplate stringRedisTemplate(@Autowired RedisConnectionFactory redisConnectionFactory) {
StringRedisTemplate stringRedisTemplate = new StringRedisTemplate();
stringRedisTemplate.setConnectionFactory(redisConnectionFactory);
return stringRedisTemplate;
}
범용적으로 사용하기 위해 객체를 json
으로 직렬화 저장하는 경우가 많은데, 이 때 StringRedisTemplate
을 사용하면 좋다.
테스트를 위한 컨트롤러를 생성
@Slf4j
@Controller
@RequestMapping("/template")
public class RedisTemplateController {
@Autowired
StringRedisTemplate stringRedisTemplate;
@GetMapping("/sample")
public void sample(@RequestParam(required = false) String key, Model model) {
log.info("sample invoked");
Set<String> keys;
if (StringUtils.hasText(key)){
keys = stringRedisTemplate.keys(key);
}
else {
keys= stringRedisTemplate.keys("*");
}
model.addAttribute("keys", new ArrayList<>(keys));
}
@GetMapping("/insert")
public String insert(@RequestParam String key, @RequestParam String value) {
log.info("insert invoked, {}:{}", key, value);
stringRedisTemplate.opsForValue().set(key, value);
stringRedisTemplate.opsForValue().get(key);
return "redirect:/template/sample";
}
@GetMapping("/deleteAll")
public String deleteAll(Model model) {
log.info("deleteAll called()....");
stringRedisTemplate.delete(stringRedisTemplate.keys("*"));
return "redirect:/template/sample";
}
}
CrudRepository
를 사용해서 redis
에 데이터를 삽입해도 되지만 StringRedisTemplate
혹은 그냥 RedisTemplate
의 메서드로도 충분이 삽입 가능하다.
<!DOCTYPE html>
<html class="no-js" lang="en"
xmlns:th="http://www.thymeleaf.org">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
sample_radis
<form action="/redis/insert">
key: <input type="text" name="key" >
value: <input type="text" name="value">
<button type="submit">submit</button>
</form>
<button onclick="location.href='/redis/deleteAll'">delete all</button>
<p th:each="p:${keys}">[[${p}]]</p>
</body>
</html>
지금까지 저장된 키 목록이 출력된다.
RedisTemplate 직렬화
@Bean
public RedisTemplate<String, Object> redisTemplate(@Autowired RedisConnectionFactory redisConnectionFactory) {
RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();
redisTemplate.setConnectionFactory(redisConnectionFactory);
redisTemplate.setKeySerializer(new StringRedisSerializer());
return redisTemplate;
}
@Bean(name = "jacksonRedisTemplate")
public RedisTemplate<String, Object> jacksonRedisTemplate(RedisConnectionFactory redisConnectionFactory) {
RedisTemplate<String, Object> template = new RedisTemplate<>();
template.setConnectionFactory(redisConnectionFactory);
GenericJackson2JsonRedisSerializer serializer = new GenericJackson2JsonRedisSerializer();
template.setKeySerializer(new StringRedisSerializer());
template.setValueSerializer(serializer);
return template;
}
ValueSerializer
를 설정하지 않거나 GenericJackson2JsonRedisSerializer
를 사용할 수 있다.
@PostMapping("/normal")
public void addUserData(@RequestBody UserData userData) {
String key = userData.getUsername();
redisTemplate.opsForValue().set(key, userData);
}
@GetMapping("/normal/{username}")
public UserData getUserData(@PathVariable String username) {
return (UserData) redisTemplate.opsForValue().get(username);
}
@PostMapping("/jackson")
public void addUserDataJackson(@RequestBody UserData userData) {
String key = userData.getUsername();
jacksonRedisTemplate.opsForValue().set(key, userData);
}
@GetMapping("/jackson/{username}")
public UserData getUserDataJackson(@PathVariable String username) {
return (UserData) jacksonRedisTemplate.opsForValue().get(username);
}
위 API 를 통해 테스트하고 redis 에 들어가 value 를 조회하면 아래와 같이 출력된다.
기본 ValueSerializer
의 경우 객체 바이트 배열로 저장된다.
��sr com.example.redis.model.UserData,�3a&�LagetLjava/lang/Integer;LdesctLjava/lang/String;Lemailq~Lnicknameq~Lusernameq~xpsrjava.lang.Integer⠤���8Ivaluexrjava.lang.Number������xptSoftware Developertjohn@example.comtjohnnytjohn_doe
GenericJackson2JsonRedisSerializer
를 사용할 경우 @class
정보를 가지는 json 객체로 저장된다.
{"@class":"com.example.redis.model.UserData","nickname":"johnny","username":"john_doe","email":"john@example.com","age":30,"desc":"Software Developer"}
@class
정보 없이 순수 json 으로 저장하기 위해 GenericJackson2JsonRedisSerializer
에 ObjectMapper
를 지정하는 경우도 있다.
GenericJackson2JsonRedisSerializer serializer = new GenericJackson2JsonRedisSerializer(objectMapper);
Redisson 분산락
Redis
기반 분산락 기능을 지원하는 라이브러리.
Pub/Sub
기반의 락을 사용하기에, 계속해서 CPU 자원을 사용하는 스핀락보다 효율적이다.
// 내부적으로 spring data redis 사용
implementation 'org.redisson:redisson-spring-boot-starter:3.30.0'
@Bean
public RedissonClient redissonClient() {
Config config = new Config();
config.useSingleServer().setAddress("redis://localhost:6379");
return Redisson.create(config);
}
@Slf4j
@Service
@RequiredArgsConstructor
public class DistributedLockService {
private final RedissonClient redissonClient;
private Integer count = 0;
public void executeWithLock(String lockKey) {
RLock lock = redissonClient.getLock(lockKey);
boolean isLocked = false;
try {
// 락을 획득하기 위해 100초 동안 시도하고, 10초 동안 락을 유지.
isLocked = lock.tryLock(100, 10, TimeUnit.SECONDS);
if (isLocked) {
// 락을 획득한 상태에서 실행할 작업
Thread.sleep(100);
count += 1;
log.info("Lock acquired. Executing protected code. counter:{}", count);
} else {
log.info("Could not acquire lock.");
}
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
if (isLocked) {
lock.unlock(); // TTL 을 넘길경우 unlock 호출시 IllegalMonitorStateException 발생
log.info("Lock released.");
}
}
}
public void executeWithoutLock() {
try {
Thread.sleep(100);
count += 1;
log.info("Lock acquired. Executing protected code. counter:{}", count);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
해당 라이브러리의 주의해야할 점은 LOCK TTL
동안 임계코드 실행이 완료되지 않는다면 중복실행될 수 있다.
TTL 로 인해 Lock 이 해제되어도 어플리케이션 코드는 지속 실행되다 lock.unlock()
시점에야 TTL 을 넘겼음을 알 수 있다.
데드락 상황을 감수하고 TTL 을 무제한으로 설정한다면 확실하게 임계영역을 지킬 수 있다.
AOP 분산락
try, catch, finally
형태의 분산락 구조를 AOP 형태로 구성하는것이 대부분.
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface DistributedLock {
String value();
}
@Slf4j
@Aspect
@Component
@RequiredArgsConstructor
public class DistributedLockAspect {
private final RedissonClient redissonClient;
@Around("@annotation(DistributedLock)")
public Object executeWithLock(ProceedingJoinPoint joinPoint) throws Throwable {
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
DistributedLock distributedLock = signature.getMethod().getAnnotation(DistributedLock.class);
String lockKey = distributedLock.value(); // 어노테이션에서 락 키 가져오기
RLock lock = redissonClient.getLock(lockKey);
log.info("Lock acquired. Executing protected code.");
try {
lock.lock();
Object result = joinPoint.proceed();
return result;
} finally {
lock.unlock();
log.info("Lock released.");
}
}
}
@DistributedLock("executeWithAopLock")
public void executeWithAopLock() {
// 비즈니스 로직 실행
log.info("business code invoked");
for (int i = 0; i < 10; i++) {
count += 1;
}
log.info("business end");
}
데모코드
Redis demo https://github.com/Kouzie/spring-boot-demo/tree/main/redis-demo