본문 바로가기
데이터베이스/Oracle

[Oracle] 언더바 포함 문자 검색하는 SQL 조건문 (ESCAPE)

by joeun 2024. 6. 2.

사용자 아이디와 같이 '_'(언더바) 특수문자를 포함한 데이터를 찾아야할 때가 있다.

 

보통 SQL에서 문자열 검색은 WHERE 조건에 LIKE 연산자와 와일드카드(%,_)를 사용하지만,

검색 문자열에 '_' 언더바 문자가 포함되어 있을 경우, 원하는 결과가 제대로 나오지 않을 것이다.

 

 


잘못된 예시

 

예를 들어 'admin_' 문자열을 포함한 관리자 ID를 검색하고자 아래 SQL문을 실행한다고 해보자.

원하는 데이터 외에 다른 데이터도 결과에 포함되는 것을 볼 수 있다. 

 

SELECT *
  FROM USER_INFO
 WHERE USER_ID LIKE '%admin_%'

 

 

실행 결과)

USER_ID USER_NAME
admin_1 관리자1
admin_2 관리자2
admin123 관리자
admini 관리자

 


 

SQL에서 언더바를 문자가 아니라 와일드카드로 인식했기 때문이다. 

언더바 와일드카드는 검색 문자의 갯수와 위치를 고정하고자 할 때 사용한다. 

따라서 위 SQL에서는 'admin' 문자 뒤에 문자 '하나'가 붙은 모든 데이터를 추출한다.

공식문서 내용 中)
An underscore (_) in the pattern matches exactly one character (as opposed to one byte in a multibyte character set) in the value.

 

 

'admin_1', 'admin_2' 처럼 언더바를 포함한 데이터 뿐만 아니라

'admin1', 'admin123', 'admini' 처럼 admin 뒤에 언더바가 아닌 문자가 붙은 데이터도 함께 결과로 가져오는 것이다. 

'admin' 데이터는 admin 뒤에 '_' 언더바에 해당하는 문자가 없기 때문에 실행 결과에 포함되지 않는다.

 

 

 

 


ESCAPE 절 

 

언더바를 와일드카드가 아닌 문자로 인식하게 하기 위해 ESCAPE 절을 사용할 수 있다.

escape character를 검색 문자열 내 와일드카드(%,_) 문자 앞에 배치하면 오라클이 와일드카드가 아니라 문자 그대로 인식한다. 

공식문서 내용 中)
You can include the actual characters % or _ in the pattern by using the ESCAPE clause, which identifies the escape character. If the escape character precedes the character % or _ in the pattern, then Oracle interprets this character literally in the pattern rather than as a special pattern-matching character. You can also search for the escape character itself by repeating it. For example, if @ is the escape character, then you can use @@ to search for @.

 

 

 

 

ESCAPE 사용 예시

 

검색 문자열 내 와일드카드 앞에 원하는 ESCAPE 문자를 넣고 LIKE 연산자 뒤에 ESCAPE 문자를 선언한다.

SELECT *
  FROM USER_INFO
 WHERE USER_ID LIKE '%admin\_%' ESCAPE '\';

 

실행 결과)

USER_ID USER_NAME
admin_1 관리자1
admin_2 관리자2

 

 


escape character는 '\' 백슬래시가 아닌 '@', '#' 등 다른 문자들도 가능하다. (영문, 숫자도 가능)

ex) ESCAPE '#',  ESCAPE '$',  ESCAPE 'k' ...

SELECT *
  FROM USER_INFO
 WHERE USER_ID LIKE '%admin@_%' ESCAPE '@';

 

 

 

 

참고로 ESCAPE 문자는 1개만 설정 가능하다.

여러 ESCAPE 문자를 설정할 경우, 아래와 같은 에러가 발생한다.

SELECT * FROM USER_INFO WHERE USER_ID LIKE '%admin\_%' ESCAPE '@,\' ;

ORA-01425: escape character must be character string of length 1 01425. 00000 - "escape character must be character string of length 1" *Cause: Given escape character for LIKE is not a character string of length 1. *Action: Change it to a character string of length 1.

 

 

 

 

 


escape character 포함 문자열 만들기(Java)

 

<!-- SELECT 문 (Mybatis) -->
<select id="searchUsersByUserId" parameterType="java.lang.String" resultType="com.example.model.User">
  SELECT * 
    FROM user_info
   WHERE user_id LIKE '%' || #{userId} || '%' ESCAPE '\'
</select>

 

public static String makeSelectStr(String input) {
        StringBuilder builder = new StringBuilder();

        for (int i = 0; i < input.length(); i++) {
            char currentChar = input.charAt(i);
            if (currentChar == '_') {
                builder.append('\\'); //백슬래시를 문자로 인식하기 위해 두 번 입력
            }
            builder.append(currentChar);
        }

        return builder.toString();
}

 

 

 

 

 

 

 


참고👀

 

LIKE

LIKE The LIKE conditions specify a test involving pattern matching. Whereas the equality operator (=) exactly matches one character value to another, the LIKE conditions match a portion of one character value to another by searching the first value for the

docs.oracle.com

 

 

like 절 주의할 점

필요하지 않은 디비 값까지 조회되는 운영이슈가 있었다. db 쿼리문 조회 시 검색조건에서 예외케이스 있음을 알게된 경험이 있다. 로그부터해서 api 까지 거꾸로 따라가다가 like 절에서 escape 처

keepgoing0328.tistory.com