본문 바로가기

MySQL

Groub By 와 having 절

GROUP BY 절이란?

GROUP BY 절은 SQL에서 특정 열의 값에 따라 행 집합을 요약된 그룹으로 나누는 데 사용된다. 이 절은 주로 집계 함수(예: SUM, AVG, MAX, MIN, COUNT 등)와 함께 사용되어, 각 그룹에 대한 요약 정보를 제공한다.

  • GROUP BY 절은 데이터들을 원하는 그룹으로 나눌 수 있다.
  • 나누고자 하는 그룹의 컬럼명을 SELECT절과 GROUP BY절 뒤에 추가하면 된다.
  • 집계 함수와 함께 사용되는 상수는 GROUP BY 절에 추가하지 않아도 된다. (많이 실수 하는 부분)

 

HAVING 절이란?

HAVING 절은 GROUP BY 절과 함께 사용되며, 특정 조건을 만족하는 그룹만을 필터링하는데 사용된다. WHERE 절이 개별 행에 대해 조건을 적용하는 반면, HAVING 절은 그룹화된 결과에 대한 조건을 적용한다. 집계 함수의 결과를 기반으로 조건을 적용할 수 있기 때문에 GROUP BY 절 이후에 사용된다.

 

학습 예제

고등학교에서 다양한 클래스의 학생들이 시험을 치렀다고 가정해 보자. 각 클래스별 평균 점수가 70점 이상인 클래스만을 찾고 싶을 때 사용할 수 있는 SQL 쿼리를 작성해 보겠다.

id
name
class
score
1
Alice
A
85
2
Bob
B
75
3
Charlie
A
65
4
David
A
70
5
Eve
B
95
6
Frank
C
88
7
Grace
C
92
8
Hannah
B
78
9
Ian
A
85
10
Jill
C
90
create database mydb3;
use mydb3; 

CREATE TABLE tb_student (
    id INT AUTO_INCREMENT PRIMARY KEY,
    name VARCHAR(100) NOT NULL,
    class CHAR(1) NOT NULL,
    score INT NOT NULL
);
    
  
INSERT INTO tb_student (name, class, score) VALUES
('Alice', 'A', 85),
('Bob', 'B', 75),
('Charlie', 'A', 65),
('David', 'A', 70),
('Eve', 'B', 95),
('Frank', 'C', 88),
('Grace', 'C', 92),
('Hannah', 'B', 78),
('Ian', 'A', 85),
('Jill', 'C', 90);

-- 전체 조회 
select * from tb_student; 

-- 1단계 : 클래스 별로 그룹화 시켜 보자. 
select class, AVG(score) AS AVERAGESCORED
from tb_student
group by class; 


-- 2단계 : 클래스별로 그룹화된 결과 집합에서 필터 적용하기 
-- having 절에 조건은 결과집이 먼저 생성된 후 기준으로 필터 조건을 설정해야 한다. !!!(주의) 
select class, AVG(score) as 평균 
from tb_student
group by class
having 평균 >= 80;

-- 문제
-- 각 클래스에서 가정 높은 점수와 가장 낮은 점수를 찾아 보세요 
select class, MAX(score) as 고득점, MIN(score) as 저득점 
from tb_student
group by class
having MAX(score) >= 93;

-- 참고  만약 낮은 버전에 db를 사용한다면 별칭의 컬럼은 적용안될 수 있다.
-- 메모리 디비 버전 낮을 수 있다. ()
CREATE TABLE tb_employees (
    employee_id INT AUTO_INCREMENT PRIMARY KEY,
    name VARCHAR(100) NOT NULL,
    department VARCHAR(50) NOT NULL,
    salary INT NOT NULL
);


INSERT INTO tb_employees (name, department, salary) VALUES
('John Doe', 'Sales', 48000000),
('Jane Smith', 'Sales', 55000000),
('Alice Johnson', 'Marketing', 50000000),
('Chris Lee', 'Marketing', 45000000),
('Bob Brown', 'HR', 35000000),
('Patricia Pink', 'HR', 40000000),
('Michael White', 'Engineering', 75000000),
('Anna Black', 'Engineering', 65000000),
('Linda Green', 'Engineering', 72000000),
('James Red', 'Engineering', 68000000),
('Larry Blue', 'Sales', 52000000),
('Jessica Purple', 'Sales', 51000000),
('Amber Yelow', 'Marketing', 47000000),
('Peter Orange', 'HR', 43000000),
('Lisa Teal', 'HR', 39000000);


SELECT * FROM tb_employees;

-- 문제 1 
-- 각 부서별 평균 급여 계산 단! 소수점 제거해주세요 
SELECT department, round(AVG(salary), 0) as 평균급여 
FROM tb_employees
GROUP BY department;


-- 문제 2 
-- 평균 급여가 50000000 금액 이상인 부서 찾아 주세요 
SELECT department, round(AVG(salary), 0) as 평균급여 
FROM tb_employees
GROUP BY department
having 평균급여 >= 50000000;



-- 문제 3 
-- 각 부서에서 가장 높은 급여를 받는 직원의 급여 금액 출력해 주세요 
SELECT department, MAX(salary)
FROM tb_employees
GROUP BY department;



-- 문제 4 
-- 특정 부서에서 근무하는 직원 수가 3명 이상인 부서만 출력해 주세요 
SELECT department, count(*) as 부서인원수 
FROM tb_employees
GROUP BY department
having 부서인원수 >= 3;
 

-- 문제 5 
-- 각 부서별 평균 급여와 직원 수를 출력해주세요 
SELECT department, avg(salary) as 평균급여,  count(*) as 부서인원수 
FROM tb_employees
GROUP BY department
728x90

'MySQL' 카테고리의 다른 글

MySQL 계정 생성 및 권한  (0) 2024.06.10
SQL 쿼리의 일반적인 처리 순서  (1) 2024.06.10
MySQL Function  (0) 2024.06.07
테이블 복사 및 데이터 추가  (0) 2024.06.07
MySQL JOIN  (0) 2024.06.05