devFancy BE Developer

[Programmers] SQL > String, Date 풀이 모음

2023-12-22
devFancy
SQL

자동차 대여 기록에서 장기/단기 대여 구분하기(Lv.1)

자동차 대여 기록에서 장기/단기 대여 구분하기

  • 문제: CAR_RENTAL_COMPANY_RENTAL_HISTORY 테이블에서 대여 시작일이

    2022년 9월에 속하는 대여 기록에 대해서 대여 기간이 30일 이상이면 ‘장기 대여’

    그렇지 않으면 ‘단기 대여’ 로 표시하는 컬럼(컬럼명: RENT_TYPE)을 추가하여

    대여기록을 출력하는 SQL문을 작성해주세요.

    결과는 대여 기록 ID를 기준으로 내림차순 정렬해주세요.

Answer Code(2023.12.22)

SELECT HISTORY_ID, CAR_ID, 
DATE_FORMAT(START_DATE, '%Y-%m-%d') AS START_DATE,
DATE_FORMAT(END_DATE,  '%Y-%m-%d') AS END_DATE,
CASE WHEN DATEDIFF(END_DATE, START_DATE) + 1 >= 30 THEN "장기 대여"
ELSE "단기 대여" END AS RENT_TYPE
FROM CAR_RENTAL_COMPANY_RENTAL_HISTORY
WHERE START_DATE BETWEEN '2022-09-01' AND '2022-09-30'
ORDER BY HISTORY_ID DESC;

문제 풀이

문제를 풀기 위해서 해야할 작업들

  1. 대여시작일이 2022년 9월인 기옥에 대해서 대여 기간이 30일 이상인 경우 “장기 대여” 그렇지 않으면 “단기 대여”로 구분(RENT_TYPE 컬럼 추가)
  2. 데이터 포맷은 예시와 동일하게 나타내기
  3. 대여 기옥 ID를 기준으로 내림차순하기

DATE_FORMAT : MySQL에서 사용되는 날짜 데이터의 출력 형식을 지정하는 함수

  • DATE_FORMAT(START_DATE, '%Y-%m-%d')은 start_date 에서 '%Y-%m-%d' 형식으로 반환해준다.

    (대부분의 날짜 데이터는 %Y-%m-%d 형식을 많이 사용한다.)

  • %Y : 4자리 연도(예: 2023) / %y : 2자리 연도 (예: 23)

  • %m: 월 (01부터 12까지)

  • %d: 일 (01부터 31까지)

DATEDIFF: MySQL에서 사용되는 두 날짜 간의 차이를 계산하는 함수

  • 이 함수는 첫 번째 날짜에서 두 번째 날짜를 뺀 결과를 반환해준다.

  • 기본 사용 형식: DATEDIFF(END_DATE, START_DATE)

  • 중요) 오늘 대여하고 오늘 반납해도 대여 기간은 하루 기간이기 때문에, (END_DATE - START_DATE) +1 >= 30 으로 계산해야 한다.

# 대여기간 예시
2023년 12월 22일 대여 >> 2023년 12월 22일 반납: 1일
= (22 - 22)일 + 1일
= 1일

2023년 12월 22일 대여 >> 2023년 12월 23일 반납: 2일
= (23 - 22)일 + 1일
= 2일

CASE 문: 조건에 따라 새로운 값을 부여하여 새로운 컬럼을 부여하는 경우에 사용한다.

  • 아래는 CASE 문의 기본 사용 형식이다.
CASE
WHEN 조건 1
THEN 조건 1 만족  반환하는 
WHEN 조건 2
THEN 조건 2 만족  반환하는 
ELSE 조건들에 만족  하는 경우 반환 
END 

특정 옵션이 포함된 자동차 리스트 구하기(Lv.1)

특정 옵션이 포함된 자동차 리스트 구하기

  • 문제: CAR_RENTAL_COMPANY_CAR 테이블에서 ‘네비게이션’ 옵션이 포함된 자동차 리스트를 출력하는 SQL문을 작성해주세요.

    결과는 자동차 ID를 기준으로 내림차순 정렬해주세요.

Answer Code1

select car_id, car_type, daily_fee, options
from car_rental_company_car
where options like '%네비게이션%'
order by car_id desc

문제 풀이

문제를 풀기 위해서 해야할 작업들

  1. 네비게이션 옵션이 포함된 자동차 리스트를 출력한다.
  2. 자동차 ID를 기준으로 내림차순으로 정렬한다.
  • CAR_RENTAL_COMPANY_CAR 테이블에서 ‘네비게이션’ 옵션이 포함 => where + like 절을 사용한다. => like '%네비게이션%'

  • order by car_id desc => DESC는 내림차순(ASC는 오름차순 이다)

조건에 부합하는 중고거래 상태 조회하기(Lv 2)

조건에 부합하는 중고거래 상태 조회하기

  • 문제: USED_GOODS_BOARD 테이블에서 2022년 10월 5일에 등록된

    중고거래 게시물의 게시글 ID, 작성자 ID, 게시글 제목, 가격, 거래상태를 조회하는 SQL문을 작성해주세요.

    거래상태가 SALE 이면 판매중, RESERVED이면 예약중, DONE이면 거래완료 분류하여 출력해주시고,

    결과는 게시글 ID를 기준으로 내림차순 정렬해주세요.

Answer Code

SELECT BOARD_ID, WRITER_ID, TITLE, PRICE, 
    CASE 
        WHEN STATUS = 'SALE' THEN '판매중'
        WHEN STATUS = 'RESERVED' THEN '예약중'
        WHEN STATUS = 'DONE' THEN '거래완료' 
        END STATUS
FROM USED_GOODS_BOARD
WHERE BOARD_ID IN(
  SELECT BOARD_ID 
  FROM USED_GOODS_BOARD
  WHERE CREATED_DATE = '2022-10-05'
)
ORDER 
    BY BOARD_ID DESC;

문제 풀이

문제를 풀기 위해서 해야할 작업들

  1. 2022년 10월 5일에 등록된 중고 거래 게시물에서 (WHERE)
  2. 필요한 컬럼들 조회(SELECT)
  3. 거래 상태에 따라 분류해서 출력(CASE)
  4. 게시글 ID를 기준으로 내림차순 정렬(ORDER BY)

조건에 맞는 사용자 정보 조회하기(Lv.3)

조건에 맞는 사용자 정보 조회하기

  • 문제: USED_GOODS_BOARDUSED_GOODS_USER 테이블에서

    중고 거래 게시물을 3건 이상 등록한 사용자의 사용자 ID, 닉네임, 전체주소, 전화번호를 조회하는 SQL문을 작성해주세요.

    이때, 전체 주소는 시, 도로명 주소, 상세 주소가 함께 출력되도록 해주시고,

    전화번호의 경우 xxx-xxxx-xxxx 같은 형태로 하이픈 문자열(-)을 삽입하여 출력해주세요.

    결과는 회원 ID를 기준으로 내림차순 정렬해주세요.

Answer Code

select user.user_id, user.nickname, 
concat(user.city,' ', user.street_address1,' ', user.street_address2) AS 전체주소,
concat(LEFT(tlno, 3), '-', MID(tlno, 4, 4),'-', RIGHT(tlno, 4)) AS 전화번호
from used_goods_user as user
join used_goods_board as board
on user.user_id = board.writer_id
group by board.writer_id
having count(board.writer_id) >= 3
order by user.user_id desc;

문제 풀이

문제를 풀기 위해서 해야할 작업들

  1. USED_GOODS_BOARD 테이블에서 writer_id 컬럼의 값을 count하여 3개 이상인 writer_id 찾기
  2. CONCAT() 함수와 USED_GOODS_USER의 CITY, STREET_ADDRESS1, STREET_ADDRESS2를 활용하여 ‘전체주소’ 생성하기
  3. CONCAT() 함수와 USED_GOODS_USER의 TLNO를 활용하여 ‘전화번호’ 생성하기
  4. USER_ID 기준으로 내림차순하기

GROUP BY 구문: 주로 집계 함수(COUNT)와 함께 사용되어 특정 열의 값에 기반하여 그룹을 형성하는 역할

group by board.writer_id
having count(board.writer_id) >= 3
  • USER_GOODS_BOARD 테이블에서 WRITER_ID 열을 기준으로 그룹을 형성한다.

  • 그런 다음, HAVING 구문을 사용하여 USER_GOOD_BOARD 테이블에서 각 작성자(WRITER_ID)가 최소 3개 이상의 게시물을 작성한 경우에만 해당 그룹이 선택된다.

CONCAT() 함수: SQL에서 사용되는 문자열을 연결하는 함수

concat(user.city,' ', user.street_address1,' ', user.street_address2) AS 전체주소
  • SELECT 절에서 “CONCAT() 함수”를 이용해서 CITY, STREET_ADDRESS1, STREET_ADDRESS2 컬럼을 하나의 문자열로 연결하는 작업이다.

  • 예시에서 나온 것과 같게 만들기 위해서는 공백(‘ ‘)으로 각 컬럼의 값을 구분하고, AS 절을 이용하여 요구한 컬럼의 이름과 동일하게 전체주소 라는 컬럼을 부여했다.

concat(LEFT(tlno, 3), '-', MID(tlno, 4, 4),'-', RIGHT(tlno, 4)) AS 전화번호
  • “CONCAT() 함수”를 이용해서 연결 사이에 ‘-‘ 으로 구분하여 하나의 문자열을 만들었다.

    • LEFT(tlno, 3): 전화번호 문자열의 왼쪽에서 3자리를 추출

    • MID(tlno, 4, 4): 전화번호 문자열에서 4번째 위치에서부터 4자리를 추출

    • RIGHT(tlno, 4): 전화번호 문자열의 오른쪽에서 4자리를 추출합니다.

다른 방법: SUBSTR((또는 SUBSTRING)) 함수 사용 : 문자열에서 일부분을 추출하는 데 사용되는 SQL 함수이다.

CONCAT(SUBSTR(TLNO, 1, 3), '-', SUBSTR(TLNO, 4, 4), '-', SUBSTR(TLNO, 8)) AS 전화번호
  • 일반적인 사용 형식: SUBSTR(string, start_position, length)

    • string: 추출하려는 문자열

    • start_position: 추출을 시작할 위치입니다. 1부터 시작한다.

    • length (선택사항): 추출하려는 부분의 길이를 지정한다. 만약 이 부분을 생략하면 시작 위치부터 문자열의 끝까지 추출하게 된다.

  • SUBSTR() (또는 SUBSTRING) 함수: 문자열에서 일부분을 추출하는 데 사용되는 SQL 함수이다.

    • SUBSTR(TLNO, 1, 3): 전화번호 문자열의 1번째 위치에서부터 3자리를 추출

    • SUBSTR(TLNO, 4, 4): 전화번호 문자열에서 4번째 위치에서부터 4자리를 추출

    • SUBSTR(TLNO, 8): 전화번호 문자열의 8번째 위치에서부터 끝까지 추출

Answer Code2

SELECT  USER_ID
        , NICKNAME
        , CONCAT(CITY, ' ', STREET_ADDRESS1, ' ', STREET_ADDRESS2) AS '전체주소'
        , CONCAT(SUBSTR(TLNO, 1, 3), '-', SUBSTR(TLNO, 4, 4), '-', SUBSTR(TLNO, 8)) AS '전화번호'
FROM    USED_GOODS_USER 
WHERE   USER_ID IN (
                     SELECT  WRITER_ID
                     FROM    USED_GOODS_BOARD
                     GROUP BY WRITER_ID
                     HAVING  COUNT(*) >= 3
                 )
ORDER 
  BY  USER_ID DESC;

문제 풀이

  • 위 정답은 다른 사람이 푼 것을 가져와봤다.

  • Answer Code1는 join을 사용했는데, Answer Code2는 join 없이 서브 쿼리를 사용한 SQL 구문이다.

조회수가 가장 높은 중고거래 게시물에 대한 첨부파일 경로를 조회하기(Lv.3)

조회수가 가장 높은 중고거래 게시물에 대한 첨부파일 경로를 조회하기

  • 문제: USED_GOODS_BOARDUSED_GOODS_FILE 테이블에서

    조회수가 가장 높은 중고거래 게시물에 대한 첨부파일 경로를 조회하는 SQL문을 작성해주세요.

    첨부파일 경로는 FILE ID를 기준으로 내림차순 정렬해주세요.

    기본적인 파일경로는 /home/grep/src/ 이며, 게시글 ID를 기준으로 디렉토리가 구분되고,

    파일이름은 파일 ID, 파일 이름, 파일 확장자로 구성되도록 출력해주세요.

    조회수가 가장 높은 게시물은 하나만 존재합니다.

Answer Code1

SELECT  CONCAT('/home/grep/src/', BOARD_ID, '/', FILE_ID, FILE_NAME, FILE_EXT) AS FILE_PATH
FROM  USED_GOODS_FILE
WHERE  BOARD_ID = (
  SELECT BOARD_ID
  FROM USED_GOODS_BOARD
  ORDER BY VIEWS DESC
  LIMIT 1
)
ORDER
  BY  FILE_ID DESC;

문제 풀이

문제를 풀기 위해서 해야할 작업들

  1. CONCAT() 함수를 이용하여 USED_GOODS_FILE 테이블에서 BOARD_ID 를 기준으로 디렉터리를 ‘/’ 으로 구분하고, 파일이름은 FILE_ID, FILE_NAME, FILE_EXT로 구성되도록 한다.
  2. 조건 WHERE 절에서 USED_GOODS_BOARD 테이블에서 가장 높은 조회수(VIEW)를 SELECT 절과 MAX를 이용해서 하나만 조회하도록 한다.
  3. FILE_ID를 기준으로 내림차순 정렬하여 결과를 출력한다.

[1]. CONCAT() 함수를 이용하여 USED_GOODS_FILE 테이블에서 BOARD_ID 를 기준으로 디렉터리를 ‘/’ 으로 구분하고, 파일이름은 FILE_ID, FILE_NAME, FILE_EXT로 구성되도록 한다.

SELECT  CONCAT('/home/grep/src/', BOARD_ID, '/', FILE_ID, FILE_NAME, FILE_EXT) AS FILE_PATH
FROM  USED_GOODS_FILE

[2]. 조건 where 절에서 USED_GOODS_BOARD 테이블에서 가장 높은 조회수(VIEW)를 SELECT 절과 ORDER BY, LIMIT절을 이용해서 하나만 조회한다.

WHERE  BOARD_ID = (
  SELECT BOARD_ID
  FROM USED_GOODS_BOARD
  ORDER BY VIEWS DESC
  LIMIT 1
)

[3]. FILE_ID를 기준으로 내림차순 정렬하여 결과를 출력한다.

ORDER
  BY  FILE_ID DESC;

Answer Code2

SELECT
    CONCAT('/home/grep/src/', FILE.BOARD_ID, '/', FILE_ID, FILE_NAME, FILE_EXT) AS FILE_PATH
FROM USED_GOODS_FILE AS FILE
LEFT JOIN USED_GOODS_BOARD AS BOARD
ON FILE.BOARD_ID = BOARD.BOARD_ID
WHERE VIEWS = (SELECT MAX(VIEWS) FROM USED_GOODS_BOARD)
ORDER BY FILE_ID DESC

조건별로 분류하여 주문상태 출력하기(Lv 3)

조건별로 분류하여 주문상태 출력하기

  • 문제: FOOD_ORDER 테이블에서 5월 1일을 기준으로 주문 ID, 제품 ID, 출고일자, 출고여부를 조회하는 SQL문을 작성해주세요.

    출고여부는 5월 1일까지 출고완료로 이 후 날짜는 출고 대기로 미정이면 출고미정으로 출력해주시고,

    결과는 주문 ID를 기준으로 오름차순 정렬해주세요.

Answer Cod(2023.12.26)

SELECT  ORDER_ID, PRODUCT_ID, DATE_FORMAT(OUT_DATE, '%Y-%m-%d') as OUT_DATE,
        case 
            when OUT_DATE <= '2022-05-01' then '출고완료' 
            when OUT_DATE > '2022-05-01' then '출고대기'
            else '출고미정'
        end  '출고여부'
from FOOD_ORDER
order by ORDER_ID asc;

문제 풀이

문제를 풀기 위해서 해야할 작업들

  1. FOOD_ORDER 테이블에서 5월 1일을 기준으로 ORDER_ID, PRODUCT_ID, OUT_DATE, 출고여부를 조회한다.
  2. 이때 출고여부는 5월 1일까지 출고완료로, 이후 날짜는 출고대기, 미정이면 출고미정으로 출력한다. -> CASE 문을 사용
  3. 예시처럼 OUT_DATE를 yyyy-mm-dd 처럼 출력하기 위해 DATE_FORMAT(OUT_DATE, ‘%Y-%m-%d’) 를 사용한다. 이때 대문자 Y는 4자리, 소문자 y는 2자리 숫자를 출력해준다.
  4. ORDER_ID 를 기준으로 오름차순 정렬해준다. -> ORDER BY ORDER_ID ASC;

대여 기록이 존재하는 자동차 리스트 구하기(Lv 3)

대여 기록이 존재하는 자동차 리스트 구하기

  • 문제: CAR_RENTAL_COMPANY_CAR 테이블과 CAR_RENTAL_COMPANY_RENTAL_HISTORY 테이블에서

    자동차 종류가 ‘세단’인 자동차들 중 10월에 대여를 시작한 기록이 있는 자동차 ID 리스트를 출력하는 SQL문을 작성해주세요.

    자동차 ID 리스트는 중복이 없어야 하며, 자동차 ID를 기준으로 내림차순 정렬해주세요.

Answer Code(2023.12.26)

SELECT DISTINCT (A.CAR_ID)
FROM CAR_RENTAL_COMPANY_RENTAL_HISTORY A
LEFT JOIN CAR_RENTAL_COMPANY_CAR B ON A.CAR_ID = B.CAR_ID
WHERE B.CAR_TYPE = '세단'
    AND SUBSTR(A.START_DATE, 6, 2) = 10 
ORDER 
    BY A.CAR_ID DESC;

문제풀이

문제를 풀기 위해서 해야할 작업들

  1. CAR_RENTAL_COMPANY_RENTAL_HISTORY 테이블과 CAR_RENTAL_COMPANY_CAR 테이블을 조인하기 -> JOIN
  2. 자동차 종류가 ‘세단’인 CAR_ID를 선택 -> WHERE
  3. 그리고 10월에 대여를 시작한 기록 -> 데이터 필터링(AND SUBSTR)
  4. CAR_ID 리스트를 출력
  5. CAR_ID 중복 제거 -> DISTINCT
  6. CAR_ID를 기준으로 내림차순 정렬
  • CAR_RENTAL_COMPANY_RENTAL_HISTORY 테이블에서 CAR_RENTAL_COMPANY_CAR 테이블을 LEFT JOIN으로 조인을 실행해줬다.

  • 기준은 두 테이블에서 CAR_ID가 동일한 컬럼과 값을 가지고 있어서 CAR_ID를 이용했다.

  • 두번째 작업은 WHERE 절에서 CAR_ID가 ‘세단’인 데이터만 조회가 되도록 조건 필터링을 넣어줬다.

  • 세번째 작업은 데이터 필터링을 위해 SUBSTR()을 이용해 ‘월’ 정보만을 가져온 다음 이 내역이 10인 데이터만 가져오도록 조건 필터링을 넣어줬다.

  • 네번째 작업은 CAR_ID 중복 제거를 위해 DISTINCT를 사용했다.

오랜 기간 보호한 동물(2)(Lv 3)

오랜 기간 보호한 동물(2)

  • 문제: 입양을 간 동물 중, 보호 기간이 가장 길었던 동물 두 마리의 아이디와 이름을 조회하는 SQL문을 작성해주세요.

    이때 결과는 보호 기간이 긴 순으로 조회해야 합니다.

Answer Code(2023.12.27)

SELECT A.ANIMAL_ID, A.NAME
FROM ANIMAL_INS A
    LEFT JOIN ANIMAL_OUTS B ON A.ANIMAL_ID = B.ANIMAL_ID
ORDER BY B.DATETIME - A.DATETIME DESC
LIMIT 2

문제 풀이

문제 접근

  1. 두 개의 테이블 조인(JOIN)
  2. 보호 기간이 가장 길었던 동물 두마리에 대한 필터링(ORDER BY, LIMIT)
  3. 아이디와 이름을 조회(SELECT)

취소되지 않은 진료 예약 조회하기(Lv 4)

취소되지 않은 진료 예약 조회하기

  • 문제: PATIENT, DOCTOR 그리고 APPOINTMENT 테이블에서

    2022년 4월 13일 취소되지 않은 흉부외과(CS) 진료 예약 내역을 조회하는 SQL문을 작성해주세요.

    진료예약번호, 환자이름, 환자번호, 진료과코드, 의사이름, 진료예약일시 항목이 출력되도록 작성해주세요.

    결과는 진료예약일시를 기준으로 오름차순 정렬해주세요.

Answer Code(2023.12.26)

SELECT A.APNT_NO,
       P.PT_NAME,
       A.PT_NO,
       A.MCDP_CD,
       D.DR_NAME,
       A.APNT_YMD
FROM APPOINTMENT A
    LEFT JOIN PATIENT P ON A.PT_NO = P.PT_NO
    LEFT JOIN DOCTOR D ON A.MDDR_ID = D.DR_ID
WHERE A.MCDP_CD = 'CS'
    AND A.APNT_CNCL_YN = 'N'
    AND LEFT(APNT_YMD, 10) = '2022-04-13'
ORDER BY
    A.APNT_YMD ASC

문제 풀이

문제 접근

  1. 3개의 테이블 조인하기(JOIN)
  2. APPOINTMENT 테이블에서 조건에 맞는 데이터 필터링(WHERE)
  3. 필요한 컬럼들 선택(SELECT)
  4. 진료 예약일시 기준으로 오름차순 정렬

[1] 3개의 테이블 조인하기

  • APPOINTMENT 테이블을 기준으로 PATIENT 테이블과 환자 번호(PT_NO)로 LEFT JOIN

  • 그리고 APPOINTMENT 테이블을 기준으로 DOCTOR 테이블과 의사 아이디(MDDR_ID)로 LEFT JOIN

[2] WHERE절에서 조건에 맞는 데이터만 조회가 되도록 3개의 조건 필터링을 걸어준다.

  • 진료과 코드가 ‘흉부외과’인 데이터 (MCDP_CD = ‘CS’)

  • 진료 예약이 취소되지 않은 데이터 (APNT_CNCL_YN = ‘N’)

  • 진료 예약일이 ‘2022-04-13’인 데이터(LEFT(APNT_YMD,10) = ‘2022-04-13’)

[3] SELECT 문에서 필요한 컬럼을 불러온다.

  • 진료예약번호(APPOINTMENT.APNT_NO),
  • 환자이름(PATIENT.PT_NAME),
  • 환자번호(APPOINTMENT.PT_NO)
  • 진료과코드(APPOINTMENT.MCDP_CD),
  • 의사이름(DOCTOR.DR_NAME),
  • 진료예약일시(APPOINTMENT.APNT_YMD)

[4] 진료 예약일시(A.APNT_YMD) 기준으로 오름차순 정렬

  • ORDER BY A.APNY_YMD ASC
  • 마지막 작업은 CAR_ID를 기준으로 내림차순 정렬을 위해 ORDER BY 절을 사용했고, 내림차순(DESC)으로 정렬했다.

자동차 대여 기록 별 대여 금액 구하기(Lv.4)

자동차 대여 기록 별 대여 금액 구하기

  • 문제: CAR_RENTAL_COMPANY_CAR 테이블과 CAR_RENTAL_COMPANY_RENTAL_HISTORY 테이블과 CAR_RENTAL_COMPANY_DISCOUNT_PLAN 테이블에서

    자동차 종류가 ‘트럭’인 자동차의 대여 기록에 대해서 대여 기록 별로 대여 금액(컬럼명: FEE)을 구하여

    대여 기록 ID와 대여 금액 리스트를 출력하는 SQL문을 작성해주세요.

    결과는 대여 금액을 기준으로 내림차순 정렬하고,

    대여 금액이 같은 경우 대여 기록 ID를 기준으로 내림차순 정렬해주세요.

Answer Code

SELECT  D.HISTORY_ID
        , ROUND(IF(D.DISCOUNT_RATE IS NULL, D.DAILY_FEE * D.DATE_DIFF, D.DAILY_FEE * D.DATE_DIFF * (100-D.DISCOUNT_RATE)*0.01),0) AS FEE
  FROM  (
        SELECT  A.*
                , B.HISTORY_ID
                , DATEDIFF(B.END_DATE,B.START_DATE) + 1 AS DATE_DIFF
                , C.DISCOUNT_RATE
          FROM  CAR_RENTAL_COMPANY_CAR A
          LEFT
          JOIN  CAR_RENTAL_COMPANY_RENTAL_HISTORY B
            ON  A.CAR_ID = B.CAR_ID
          LEFT
          JOIN  CAR_RENTAL_COMPANY_DISCOUNT_PLAN C
            ON  CASE WHEN DATEDIFF(B.END_DATE,B.START_DATE) BETWEEN 7 AND 29 THEN C.PLAN_ID = 10
                     WHEN DATEDIFF(B.END_DATE,B.START_DATE) BETWEEN 30 AND 89 THEN C.PLAN_ID = 11
                     WHEN DATEDIFF(B.END_DATE,B.START_DATE) >= 90 THEN C.PLAN_ID = 12 END
         WHERE  A.CAR_TYPE = '트럭'
         ) D 
 ORDER
    BY   FEE DESC 
         , HISTORY_ID DESC

문제 풀이

문제를 풀기 위해서 해야할 작업들

  1. 자동차 종류가 ‘트럭’인 자동차 필터링 작업
  2. 자동차 대여 일자 계산하기
  3. 대여 일자에 맞게 할인률 반영하여 JOIN 하기
  4. 일일 대여 금액, 대여 일자, 할인율을 반영한 대여 금액 계산하기
  5. 대여 금액 내림차순, 대여 기록 ID 내림차순 정렬
  • 첫 번째 작업을 해결하기 위해 CAR_RENTAL_COMPANY_CAR 테이블에 CAR_TYPE(자동차 종류)이 ‘트럭’인 자동차만 선택하는 쿼리를 작성한다.

[1] WHERE A.CAR_TYPE = ‘트럭’

  • 두 번째 작업은 자동차 대여 일자 계산하는 것인데, START_DATE 부터 END_DATE 까지 대여한 일자를 구하기 위해

  • DATEDIFF(B.END_DATE, B.START_DATE) + 1을 사용해서 쿼리를 작성했다.

  • 1을 더해준 이유는 하루만 대여를 한 경우에도 0일이 아닌 1일이 되도록 보장하기 위해서이다.

[2] DATEDIFF(B.END_DATE,B.START_DATE) + 1 AS DATE_DIFF

  • 세 번째 작업은, 대여 일자에 맞게 할인율을 반영하여 JOIN 하는 과정이다.

  • 대여 일자를 DATE_DIFF로 생성했고, CASE 구문내에서 DATE_DIFF를 통해 대여 일수에 따라 PLAN_ID를 매칭하여 LEFT JOIN 하도록 쿼리를 작성했다.

  • 네 번째 작업은, 일일 대여 금액, 대여 일자, 할인율을 반영한 대여 금액(FEE)를 계산하는 것이다.

  • LEFT JOIN으로 할인율을 적용해서 본 결과, 할인이 적용되지 않는 데이터에는 할인율이 NULL 값으로 들어가 있다.

  • 그래서 IF와 IS NULL 함수를 통해 아래와 같이 계산하도록 쿼리를 작성한다.

    • 할인율이 NULL 값인 경우: 일일 대여 금액 * 대여 일자 = 대여 금액(FEE)

    • 할인율이 있는 경우: 일일 대여 금액 * 대여 일자 * (100 - 할인율) * 100 = 대여 금액(FEE)

SELECT  D.HISTORY_ID
                , ROUND(IF(D.DISCOUNT_RATE IS NULL, D.DAILY_FEE * D.DATE_DIFF
                                    , D.DAILY_FEE * D.DATE_DIFF * (100-D.DISCOUNT_RATE)*0.01),0) AS FEE
  • 마지막으로, 문제에서 요구한 정렬 조건과 같이 ORDER BY를 이용한다.
ORDER
       BY   FEE DESC 
               , HISTORY_ID DESC

Reference

  • https://kkw-da.tistory.com/category/Skill%20Sets/SQL

Index