2025년 3월 2일

[CASE WHEN THEN END] 프로그래머스 - 언어별 개발자 분류하기

cleanUrl: /CASE-WHEN-THEN-END-프로그래머스-언어별-개발자-분류하기

문제 설명

DEVELOPERS 테이블에서 GRADE별 개발자의 정보를 조회하려 합니다. GRADE는 다음과 같이 정해집니다.
  • A : Front End 스킬과 Python 스킬을 함께 가지고 있는 개발자
  • B : C# 스킬을 가진 개발자
  • C : 그 외의 Front End 개발자
GRADE가 존재하는 개발자의 GRADE, ID, EMAIL을 조회하는 SQL 문을 작성해 주세요.
결과는 GRADE와 ID를 기준으로 오름차순 정렬해 주세요.
 
 

예제 입력/출력

SKILLCODES

NAME
CATEGORY
CODE
C++
Back End
4
JavaScript
Front End
16
Java
Back End
128
Python
Back End
256
C#
Back End
1024
React
Front End
2048
Vue
Front End
8192
Node.js
Back End
16384

DEVELOPERS

ID
FIRST_NAME
LAST_NAME
EMAIL
SKILL_CODE
D165
Jerami
Edwards
jerami_edwards@grepp.co
400
D161
Carsen
Garza
carsen_garza@grepp.co
2048
D164
Kelly
Grant
kelly_grant@grepp.co
1024
D163
Luka
Cory
luka_cory@grepp.co
16384
D162
Cade
Cunningham
cade_cunningham@grepp.co
8452

출력

GRADE
ID
EMAIL
A
D162
cade_cunningham@grepp.co
A
D165
jerami_edwards@grepp.co
B
D164
kelly_grant@grepp.co
C
D161
carsen_garza@grepp.co
 
 

문제 풀이

  • SKILL_CODE의 비트 연산 활용
    • SKILL_CODE는 각 비트(bit)마다 특정 스킬을 의미하는 2의 제곱수로 저장되어 있다.
    • 특정 개발자가 특정 스킬을 가졌는지 확인하려면 비트 AND 연산자(&)를 사용하면 된다.
      • SKILL_CODE & 특정 스킬의 CODE > 0 → 해당 스킬을 보유한 개발자
      비트 연산과 SUM(CODE)의 역할
      SUM(CODE)를 사용하는 이유는 한 범주 또는 특정 조건에 해당하는 모든 스킬의 비트 값을 하나의 숫자로 결합하기 위해서 사용한다.
      예를 들어, SKILLCODES 테이블이 다음과 같다고 가정해 보자.
      NAME
      CATEGORY
      CODE (10진수)
      CODE (2진수)
      JavaScript
      Front End
      16
      00010000
      React
      Front End
      2048
      100000000000
      Vue
      Front End
      8192
      10000000000000
      Python
      Back End
      256
      100000000
      C#
      Back End
      1024
      10000000000
      1️⃣ SUM(CODE) 없이 개별 조회할 경우
      • 만약 Front End 스킬을 확인하려면, JavaScript, React, Vue 등의 개별 CODE를 모두 AND 연산으로 확인해야 한다.
      • 즉, 개발자의 SKILL_CODE가 각 스킬에 대해 다음을 만족하는지 확인해야 한다.
        • d.SKILL_CODE & 16 > 0 OR d.SKILL_CODE & 2048 > 0 OR d.SKILL_CODE & 8192 > 0
      2️⃣ SUM(CODE)를 사용하여 비트마스크 생성
      • SUM(CODE)를 사용하면 비트 연산을 한 번만 수행하면 되기 때문에 실행 속도가 빨라지고 가독성이 좋아진다는 장점이 있다.
      • 또한, 새로운 Front End 기술이 추가되더라도 쿼리를 수정할 필요 없이 SKILLCODES 테이블만 업데이트하면 된다.
        • SELECT SUM(CODE) FROM SKILLCODES WHERE CATEGORY = 'Front End';
          16 (JavaScript) + 2048 (React) + 8192 (Vue) = 10256
          d.SKILL_CODE & 10256 > 0
      • 이렇게 하면 Front End 스킬 중 하나라도 보유한 개발자를 빠르게 찾을 수 있다.
       
  • CASE 문으로 GRADE 분류
    • CASE 문을 활용하여 GRADE를 부여
      • CASE case_value WHEN when_value THEN statement_list [WHEN when_value THEN statement_list] ... [ELSE statement_list] END CASE
    • A 등급 → Front End 스킬 & Python 스킬 보유
    • B 등급 → C# 스킬 보유
    • C 등급 → Front End 스킬만 보유
 
  • HAVING을 이용한 필터링
    • CASE 문에서 어느 조건에도 해당되지 않는 경우 GRADE 값이 NULL
 

풀이 코드

WITH FRONT_SKILL AS ( SELECT CODE AS CODE FROM SKILLCODES WHERE CATEGORY = 'Front End' ), PYTHON_SKILL AS ( SELECT CODE AS CODE FROM SKILLCODES WHERE NAME = 'Python' ), CSHARP_SKILL AS ( SELECT CODE AS CODE FROM SKILLCODES WHERE NAME = 'C#' ) SELECT CASE WHEN (d.SKILL_CODE & (SELECT CODE FROM FRONT_SKILL)) > 0 AND (d.SKILL_CODE & (SELECT CODE FROM PYTHON_SKILL)) > 0 THEN 'A' WHEN (d.SKILL_CODE & (SELECT CODE FROM CSHARP_SKILL)) > 0 THEN 'B' WHEN (d.SKILL_CODE & (SELECT CODE FROM FRONT_SKILL)) > 0 THEN 'C' END AS GRADE, d.ID, d.EMAIL FROM DEVELOPERS AS d HAVING GRADE IS NOT NULL ORDER BY GRADE ASC, d.ID ASC;
 

참고 자료

MySQL :: MySQL 8.4 Reference Manual :: 15.6.5.1 CASE Statement