2022-kkogkkog
2022-kkogkkog copied to clipboard
[BE] 인덱스 설정 및 개선사항
설명
- 인덱스를 통해 조회 성능을 개선해보자
- MySQL DB에 테이블별로 100만개의 데이터를 저장하고 서비스에서 실행하는 각종 쿼리들을 실행해보자
- 인덱스 설정 전/후에 따른 실행계획 및 실제 소요시간의 차이를 비교해보자
최종적으로 설정한 인덱스 내역
data:image/s3,"s3://crabby-images/05dc8/05dc8711397d165481a00374608feb5a4b7797dd" alt="image"
data:image/s3,"s3://crabby-images/8209c/8209cdb2a7bca1b994a333ff15b6089311359790" alt="image"
CREATE INDEX coupon_sender_meeting_date ON coupon (sender_member_id, meeting_date);
CREATE INDEX coupon_receiver_meeting_date ON coupon (receiver_member_id, meeting_date);
CREATE INDEX coupon_history_coupon_id ON member_history (coupon_id);
CREATE INDEX coupon_history_host_member_id ON member_history (host_member_id);
coupon 테이블
findAllBySender 쿼리 성능 개선
핵심: sender_member_id
컬럼 단독 활용
-
from coupon join member on coupon.sender_member_id = member.id
-
where coupon.sender_member_id = 1
그러므로 sender_member_id
컬럼에 대해 단독으로 인덱스를 설정하거나, (sender_member_id, meeting_date)
순서로 인덱스를 함께 설정
- 성능 개선: 430배 개선 (0.776초 => 0.0018초)
data:image/s3,"s3://crabby-images/89004/89004aac5828dbeeeb581c4a4227bc1b131ac258" alt="image"
- 실행계획 변화
data:image/s3,"s3://crabby-images/01046/010463eb6f6b9ea37d5e62ef90c2acdb9fa2227f" alt="image"
data:image/s3,"s3://crabby-images/5fd9a/5fd9a1da8381a9443c2ff34cb366989b0ef6dfcc" alt="image"
findAllByReceiver 쿼리 성능 개선
핵심: receiver_member_id
컬럼 단독 활용
위와 동일하므로 생략
findAllByMemberAndMeetingDate 쿼리 성능 개선
핵심: sender_member_id
, receiver_member_id
, meeting_date
활용
-
meeting_date
조건을 단독으로 사용되는 쿼리는 없음. -
sender_member_id
와receiver_member_id
는 위의 쿼리들에서 단독으로 활용됨.
from coupon
join member as member1 on coupon.sender_member_id = member1.id
join member as member2 on coupon.receiver_member_id = member2.id
where (coupon.receiver_member_id = 2 or coupon.sender_member_id = 1)
and coupon.meeting_date >= '2023-01-01 00:00:00'
인덱스 설정 전: SLOW
기본 소요시간: 0.760초
data:image/s3,"s3://crabby-images/4fc2b/4fc2b247f3e425f983905e77fc9a6daec3896040" alt="image"
실행계획: coupon 테이블에 대해 Full Table Scan 실행 => BAD!
data:image/s3,"s3://crabby-images/53408/5340877353fb03fd641b1dfde2a3b2039dbce5d6" alt="image"
data:image/s3,"s3://crabby-images/c7555/c7555bbbadf26e44bc801529b5d19561c414c755" alt="image"
개별 인덱스를 단독으로 설정하는 방법: GOOD
성능개선: 230배 개선 (0.760초 => 0.0033초)
data:image/s3,"s3://crabby-images/18b95/18b95413422a758ce13a00a79ed7f0b5445d9a42" alt="image"
실행계획: 3가지 단독 인덱스에 대해 INDEX MERGE 발생
data:image/s3,"s3://crabby-images/85163/851635b83a90769fdedeb7c1bae19ea5558af0e5" alt="image"
data:image/s3,"s3://crabby-images/5ec10/5ec10a2935ad09a940dedd58afd3e6a5f70c45c7" alt="image"
복수 컬럼에 대해 인덱스 설정: GOOD
아래와 같이 쌍으로 등록하는 경우, 기본적으로 인덱스를 단독으로 등록한 경우에 비해 조금 더 개선되는 것으로 보임.
성능개선: 361배 개선 (0.760초 => 0.0021초)
create index coupon_sender_meeting_date ON coupon (sender_member_id, meeting_date);
create index coupon_receiver_meeting_date ON coupon (receiver_member_id, meeting_date);
data:image/s3,"s3://crabby-images/176b9/176b93ec25e699dbcd1865e109c453b173cb0daa" alt="image"
실행계획: 2가지 인덱스에 대해 INDEX MERGE 발생
data:image/s3,"s3://crabby-images/8722e/8722e6dc6923defc6d8a185d17ba600618a56199" alt="image"
data:image/s3,"s3://crabby-images/e7b54/e7b546805d424b8a126a42390dfe8478f8467201" alt="image"
순서를 잘못 설정하는 경우: No Effect
인덱스 컬럼 쌍에서 meeting_date 기준으로 정렬한 이후에 sender_member_id 기준으로 정렬하는 경우, 인덱스를 타지 않음.
create index coupon_meeting_date_sender ON coupon (meeting_date, sender_member_id);
실행계획: Full Table Scan인 점은 동일하며, filtered 비율 감소 (13% => 3.3%)
data:image/s3,"s3://crabby-images/1dfd3/1dfd3fae0990aecc2fcb699a85044901856d5f71" alt="image"
data:image/s3,"s3://crabby-images/f36ec/f36ec0a7020052171de908365f11f3731b29d5c7" alt="image"
Low Cardinaliy 이슈
- cardinality가 낮은
coupon_status
컬럼에 대해 인덱스를 설정하는 것은 쿼리 성능에 유의미한 차이가 없음!
from coupon
inner join member on coupon.receiver_member_id = member.id
where coupon.receiver_member_id = 2
and coupon.coupon_status = 'READY'
receiver_member_id 컬럼에만 인덱스가 설정된 경우
data:image/s3,"s3://crabby-images/27e75/27e75936a9eedb31c395e57b82fe4c663c654e65" alt="image"
coupon_status 컬럼에도 인덱스를 설정하는 경우
data:image/s3,"s3://crabby-images/2e77e/2e77e8d6c7ebc0822dc64a3ce65a37dc170c6814" alt="image"
member_history 테이블
findAllByCouponIdOrderByCreatedTimeDesc
- coupon_id 컬럼에 대해 인덱스 설정 필요.
from member_history
left outer join coupon on member_history.coupon_id = coupon.id
where coupon.id = 1
성능개선: 460배 개선 (0.783초 => 0.0017초)
data:image/s3,"s3://crabby-images/9650f/9650ff458d0b415d8ffef3eeac4571da8d29e2a1" alt="image"
실행계획 변화
data:image/s3,"s3://crabby-images/f25c9/f25c9897dcb9c5c62052c0f4550b542a07f00561" alt="image"
data:image/s3,"s3://crabby-images/66705/6670519586f1dd8d7ce9d5bba84bfc0bbd30c279" alt="image"
data:image/s3,"s3://crabby-images/8e956/8e956d6bae9e0f28e1480d1db058b9e842222305" alt="image"
findAllByHostMemberOrderByCreatedTimeDesc
- host_member_id 컬럼에 대해 인덱스 설정 필요.
성능개선: 240배 개선 (0.609 => 0.0025)
data:image/s3,"s3://crabby-images/39dbb/39dbb45c674050bcf259cbaf6469320a8096efc6" alt="image"
실행계획 변화
data:image/s3,"s3://crabby-images/11e44/11e44bebce09ce2e4d28d345670f457825e11902" alt="image"
data:image/s3,"s3://crabby-images/65b24/65b24d49569c117504701a954584f542d7ed9818" alt="image"