아직 배우는 단계라서 정확하지 않을 수 있습니다!
피드백 해주시면 참고하겠습니다 :)
추후에 구글링 및 더 많은 공부를 더 진행한 후 깨끗히 리팩토링 진행하도록 하겠습니다.
오늘은 Redis를 이용해서 포인트 랭킹을 구현했습니다.
Redis를 단순 저장의 목적으로 사용한 것 같아서 마음에 쏙 들지는 않지만 일단 공부한다고 생각하고 진행해보았습니다.
먼저 랭킹 기획 방향은 이렇게 생각했습니다.
1. 처음에 회원가입을 한다면, Point 도메인을 만들고 MySQL 데이터베이스에 포인트 저장하고 Redis ZSet에 삽입
2. 포인트 조회시 Redis를 통해 빠르게 조회하고 업데이트시에도 마찬가지로 처리
3. 주기적으로 MySQL에 백업
이렇게 생각했는데 2번까지만 해보겠습니다.
Point Domain
package yoon.community.domain.point;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.OneToOne;
import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.NoArgsConstructor;
import org.hibernate.annotations.OnDelete;
import org.hibernate.annotations.OnDeleteAction;
import yoon.community.domain.common.EntityDate;
import yoon.community.domain.member.Member;
@Getter
@AllArgsConstructor
@NoArgsConstructor
@Entity
public class Point extends EntityDate {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@OneToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "member_id", nullable = false)
@OnDelete(action = OnDeleteAction.CASCADE)
private Member member;
@Column(nullable = false)
private int point;
public Point(Member member) {
this.member = member;
this.point = 0;
}
}
먼저 도메인은 이렇게 설정해주었습니다.
도메인 메서드 같은 경우는 추후에 데이터 백업처리 부분을 만들 때 만들 예정입니다.
@Transactional(readOnly = true)
public List<PointRankingRedisResponseDto> findPointsRankingWithRedis() {
ZSetOperations<String, String> stringStringZSetOperations = redisTemplate.opsForZSet();
Set<ZSetOperations.TypedTuple<String>> typedTuples = stringStringZSetOperations.reverseRangeWithScores(RANKING_KEY, 0, 5);
List<PointRankingRedisResponseDto> result = typedTuples.stream()
.map(i -> new PointRankingRedisResponseDto().toDto(i))
.collect(Collectors.toList());
return result;
}
@Transactional
public void updatePoint(String username) {
ZSetOperations<String, String> zSetOperation = redisTemplate.opsForZSet();
zSetOperation.incrementScore(RANKING_KEY, username, 5);
}
랭킹 조회 부분 코드입니다.
Sorted Set을 사용했습니다.
원래 Set 자료구조는 중복이 허락되지 않고, 무순서라는 특징이 있습니다.
하지만 Redis에서 Sorted Set은 중복되지는 않지만 Score를 가지고 있어서 이 가중치에 따라 정렬해서 저장할 수 있습니다.
지금은 updatePoint 메서드 부분에 username을 받아서 처리했지만, 조만간 게시글 작성, 일일 출석, 댓글 등에 연관 지어서 포인트를 올려주는 방향으로 생각하고 있습니다.
이렇게 하기 위해서 일단 기초 뼈대를 더 잡아두고 분리를 해야할 것 같습니다.
Redis-cli 에서 Point가 잘 나오는 것을 확인할 수 있습니다.
'Project > Community' 카테고리의 다른 글
[커뮤니티 #17] 기본 키 타입, 네이밍 리팩토링 작업 (0) | 2023.01.01 |
---|---|
[커뮤니티 #16] 도메인 단위 테스트를 만들어보자 (1) | 2022.12.26 |
[커뮤니티 #15] 클린코드 리팩토링 (Admin API) / 도메인 메서드를 사용하는 이유 (1) | 2022.12.26 |
[커뮤니티 #14] 기존 커뮤니티 프로젝트 클린코드 리팩토링 (Comment, Report) (0) | 2022.12.21 |
[커뮤니티 #13] 기존 프로젝트 리팩토링 계획 및 진행하기 (0) | 2022.11.23 |