Spring Data Redis로 레디스 연동하는 방법
개요
스프링에는 Redis와 연동하는 여러가지 방법이 존재하는데 Data JPA와 비슷하게 Data Redis라는 라이브러리를 제공한다. Lettuce와 Jedis 2가지 구현체를 통해 통신할 수 있는 방법이 있는데, Lettuce는 별다른 추가 설정없이 사용할 수 있지만 Jedis는 추가적인 의존성이 필요하다. 요즘에는 아래의 이유로 인해 Jedis보다 Lettuce를 쓰는 추세라고 한다.
- Lecttue는 비동기로 요청을 처리하기 때문에 성능적인 이점이 있음
- Lecttue는 Jedis와는 다르게 별도의 풀을 생성하지 않아도 되기 때문에 개발이 단순함
- Jedis는 Thread-safe하지 않고 Spring boot 2.0부터 deprecated됨
본 포스팅에서는 Data Redis를 통해 스프링에서 Redis와 통신하는 방법에 대해 기술한다.
Data Redis에서는 RedisTemplate, RedisRepository 총 2가지의 방법을 제공한다. 설정부터 2가지의 방법을 하나하나 살펴보자.
설정 및 의존성 설치
spring:
redis:
host: localhost
port: 6379
먼저 application.yml파일을 열고 아래의 라인을 추가한다. 설정은 다른 방법을 사용할수도 있지만 본 포스팅에서는 최대한 간단한 설정을 위해 yml파일에 작성했다.
implementation 'org.springframework.boot:spring-boot-starter-data-redis'
다음으로 build.gradle을 열고 dependencies부분에 data-redis를 추가해준다.
RedisTemplate
@Configuration
public class RedisConfig {
@Value("${spring.redis.host}")
private String host;
@Value("${spring.redis.port}")
private int port;
@Bean
public RedisConnectionFactory redisConnectionFactory() {
return new LettuceConnectionFactory(host, port);
}
@Bean
public RedisTemplate<?, ?> redisTemplate() {
RedisTemplate<byte[], byte[]> redisTemplate = new RedisTemplate<>();
redisTemplate.setConnectionFactory(redisConnectionFactory());
return redisTemplate;
}
}
먼저 RedisConfig라는 클래스를 하나 생성해준다. 위에서 application.yml에 레디스 관련 host, port를 넣어줬기 때문에 @Value를 통해 해당 값을 가져와서 채워주기로 한다. 위 설정은 RedisRepository가 아닌 RedisTemplate를 통해 레디스와 통신하기 위한 설정값이다.
String 타입
@Service
@RequiredArgsConstructor
public class HealthCheckService {
private final RedisTemplate<String, String> redisTemplate;
public void testRedisString() {
ValueOperations<String, String> operation = redisTemplate.opsForValue();
String key = "string";
operation.set(key, "test");
operation.get(key);
System.out.println(key);
}
}
String타입은 opsForValue() 메소드를 사용하며 set(), get() 등의 메소드를 지원한다.
Set 타입
@Service
@RequiredArgsConstructor
public class HealthCheckService {
private final RedisTemplate<String, String> redisTemplate;
public void testRedisSet() {
SetOperations<String, String> operation = redisTemplate.opsForSet();
String key = "set";
operation.add(key, "h", "i", "d", "e");
Set<String> members = operation.members(key);
System.out.println(members);
}
}
Set타입은 opsForSet() 메소드를 사용하며 add(), members() 등의 메소드를 지원한다.
Hash 타입
@Service
@RequiredArgsConstructor
public class HealthCheckService {
private final RedisTemplate<String, String> redisTemplate;
public void testHash() {
HashOperations<String, Object, Object> operation = redisTemplate.opsForHash();
String key = "hash";
operation.put(key, "hide", "test");
Object value = operation.get(key, "hide");
System.out.println(value);
}
}
Hash타입은 opsForHash() 메소드를 사용하며 get(), put() 등의 메소드를 지원한다.
RedisRepository
RedisRepository를 사용하면 객체를 손쉽게 Hash타입의 자료구조로 변환하여 사용할 수 있다.
import lombok.Getter;
import org.springframework.data.annotation.Id;
import org.springframework.data.redis.core.RedisHash;
@Getter
@RedisHash(value = "user")
public class RedisUser {
@Id
private Long id;
private String name;
public RedisUser(Long id, String name) {
this.id = id;
this.name = name;
}
}
먼저 위처럼 사용할 객체를 만든다. 위 예제에서는 간단하게 id, name이라는 2개의 필드만을 가진 RedisUser라는 클래스를 만들었다. @RedisHash의 value값에 특정한 값을 넣어줌으로써 추후 키를 생성할 때 prefix를 지정해줄 수 있다. 또한 @Id 어노테이션을 사용하여 prefix:구분자 형태로 값을 구분한다. 참고로 @Id 어노테이션은 코드에 나와있는 것 처럼 org.springframework.data.annotation.Id를 사용해야 한다.
만약 TTL을 주고 싶다면 RedisHash에 timeToLive옵션에 정수값을 넣어주자
public interface UserRedisRepository extends CrudRepository<RedisUser, Long> {
}
다음으로 CrudRepository를 상속받은 Repository 인터페이스를 만들어준다. 이 부분은 Data Jpa와 완전히 동일하다.
@Service
@RequiredArgsConstructor
public class HealthCheckService {
private final UserRedisRepository userRedisRepository;
public void createUser() {
RedisUser user = new RedisUser(1L, "hide");
userRedisRepository.save(user);
}
}
저장 또한 Data Jpa와 마찬가지로 객체를 생성하고 Repository의 save() 메소드를 통해 저장할 수 있다. 저장 후 Redis에서 키를 살펴보면
127.0.0.1:6379> keys *
1) "user"
2) "user:1"
위처럼 키가 생성된것을 알 수 있다. 위에서 해시로 저장할 RedisUser 객체를 생성할 때 @RedisHash의 value값을 user로 줬고 id부분에 @Id 어노테이션을 붙여줬기 때문에 prefix:구분자 형태로 저장되었다.
127.0.0.1:6379> hgetall user:1
1) "_class"
2) "slido.slidoclone.health.service.RedisUser"
3) "id"
4) "1"
5) "name"
6) "hide"
Hash형태로 저장이 되므로 hgetall을 통해 값을 살펴보면 정상적으로 저장이 되었음을 확인할 수 있다.
@Service
@RequiredArgsConstructor
public class HealthCheckService {
private final UserRedisRepository userRedisRepository;
public void getUser() {
Optional<RedisUser> user = userRedisRepository.findById(1L);
System.out.println(user.get());
}
}
값을 가져올 때도 기존 Data Jpa와 마찬가지로 동일한 메소드를 사용하면 된다. 위에서는 id값을 통해 가져오기 위해 findById() 메소드를 사용했다. 해당 값을 출력시켜보면,
slido.slidoclone.health.service.RedisUser@3cf29300
정상적으로 출력됨을 확인할 수 있다.