MySQL NULL 처리 삽질

최근에는 계속 삽질의 연속입니다. 이번 글은 MySQL에서 Null 처리 관련 삽질에 대한 내용입니다. MySQL Document에서 NULL에 대한 정의는 다음과 같습니다.

Conceptually, NULL means a missing unknown value and it is treated somewhat differently from other values.

마지막에 있는 "일반적인 값과는 다르게 처리된다" 이 부분에 주의해야 합니다.

Not equals 연산과 Null

id col1
1 A
2 A
3 B
4 null

위와 같은 데이터에 대해 다음과 같은 SQL 을 실행하면 결과가 어떻게 나타날까요?

select col2 from t1 where col1 != 'A';

필자는 3, 4번 레코드가 반환되기를 기대했지만 결과는 3번 레코드만 반환되었습니다.  DB에서 NULL이 일반적인 값과는 다르게 취급된다는 것을 알고는 있었지만 순간의 실수가 이런 문제를 발생시켰습니다. 에러를 발생시키기 아주 좋은 먹이감입니다.

일반적으로 null 여부에 대한 확인을 하기 위해서는 is null 또는 is not null 연산을 사용하지만 위의 경우에서는 요구사항이

col1의 값이 'A' 가 아닌 것 모두 레코드

이기 때문에 요구사항 그래도 SQL로 변환하면서 문제를 발생시킨 SQL을 만들게 되었습니다.

해결 방법1

이 문제를 해결하기 위해서는 가장 간단한 방법은 다음과 같이 is null을 추가하는 것입니다.

select col2 from t1 where col1 != 'A' or col1 is null;

위 예제의 경우 설명을 쉽게하기 위해 SQL 중 일부만 표현한 것인데 프로그램에서 사용되는 SQL은 아주 복잡한 경우도 많습니다. 그리고 OR 연산이 나오게 되면 연산자 우선 순위도 생각해야 하기 때문에 () 로 묶어 주는 등 where 절이 복잡해 지는 단점이 있습니다.

해결 방법2

두번째 해결 방법은 COALESCE(또는 IFNULL) 함수를 사용하는 것입니다. COALESCE 함수는 첫번째 인자가 NULL 이 아니면 그 값을 반환하고 NULL이면 두번째 파라미터로 반환하는 함수입니다. 이 함수를 이용하여 다음과 같이 사용할 수 있습니다.

select col2 from t1 where coalesce(col1, 'B') != 'A';

이 질의의 단점은 col1에 index가 있는 경우에도 index를 사용할 수 없는 단점이 있습니다.

마치며

NULL 항상 조심하세요.


Popit은 페이스북 댓글만 사용하고 있습니다. 페이스북 로그인 후 글을 보시면 댓글이 나타납니다.