[MSSQL] 로우 합치기 STUFF, FOR XML PATH, mysql group_concat 기능


개발을 하다보면 가끔씩 필요한 기능이다.


중복되는 로우가 있는데 조인하면 다른 데이터들도 같이 나올때


중복되는거 합치고 여러로우로 된 데이터들만 한 컬럼에 한줄로 표현하고 싶을때가 있다.



ID 

data 

 사과

배 

 2

망고 

사과 

배 


이런 경우


ID 

data 

사과,배 

 2

망고,사과,배 


이렇게 뽑아내고 싶을때 사용하는 방법이다.



여기서 사용할건 STUFF와 FOR XML PATH다


간단하게 그냥 예제를 보도록 하자


1
2
3
4
5
6
7
8
9
10
11
12
13
SELECT 
    DISTINCT ID, -- id 중복 제거
    STUFF(( 
        SELECT 
            ',' + DATA
        FROM 
            dataTbl b 
        WHERE 
            b.ID = a.ID 
            FOR XML PATH('') ),1,1,'') AS NAME 
FROM 
    dataTbl a
 
cs


이런식으로 사용하면 된다.


dataTbl이 단일 테이블이 아니라 그냥 괄호로 묶어서 조인해놓은 서브쿼리를 사용해도 무방하다.




MariaDB의 외래키 설정을 하던중 처음 보는 에러 메세지가 나타났다


mysql foreign key constraint is incorrectly formed



외래키가 올바르지 않다 뭐 이런 의미인것 같아 저 에러창이 떴던 이유를 찾아보니


1. 한쪽은 not Null 인데 한쪽은 null 허용일때


다시말해 대상이 되는 키가 null이면 참조할때 문제가 발생할수 있기 때문에 나타나는 에러이다.



2. 참조되어지는 컬럼이 PK,unique가 아닐때

 외래키로서의 기능에 에러가 날수 있기 때문에 저 에러 메세지가 뜬다고 한다.


생각해보면 맞는 말이긴 하다.



 

[MySql] , [MaraiaDB] 한로우에 있는 값을 구분자를 이용해 여러 로우로 출력하는 방법.



text_column 

개발자,연봉,올려줘,야근,하기,싫어



한 컬럼 한 로우에 값이 위와 같이 있을떄


기본적으로 출력을 하면 하나의 로우로 출력을 하게 된다.


만약 저 택스트를


text_column 

개발자 

연봉 

올려줘 

야근 

하기 

싫어 




위와 같은 여러줄의 로우로 출력을 하고 싶을때가 있다.


그럴때 사용하는 쿼리는 아래와 같다.


1
2
3
4
5
6
7
8
9
10
11
12
 
select
   SUBSTRING_INDEX (SUBSTRING_INDEX(테이블명.컬럼영,'구분자',numbers.n),'구분자',-1) 컬럼명
          
from 
   (select  1 n union  all  select 2  
    union  all  select  3  union  all select 4 
    union  all  select  5  union  all  select  6
    union  all  select  7  union  all  select  8 
    union  all  select  9 union  all  select  10) numbers INNER  JOIN 테이블명
    on CHAR_LENGTH ( 테이블명 . 컬럼영 ) 
      - CHAR_LENGTH ( REPLACE ( 테이블명 . 컬럼영 ,  '구분자' ,  '' ))>= numbers . n-1
cs



방법을 몰라서 구글링하고 스택오버플로우를 뒤지다가 발견한 쿼리다.



답변을 달아준 peterm 에게 감사한다.


왜 위와 같은 쿼리를 사용했을때 로우를 나눠서 출력해주는지는 모르겠으나


추가적으로 발견한건


union all select 숫자 << 이 유니온 올 해준 개수만큼 구분자로 구별해 로우로 만들어 준다는 것을 알아냈다.


만약 유니온올을 3개만 했을땐  위의 결과값이


개발자

연봉

올려줘


까지 밖에 나오지 않는다.


아마도 유니언 all을 해서 여러 로우를 임의로 만들어 준다음


구분자로 분리시킨다음 해당 로우에 순서대로 넣어 주는것 같다.


저렇게 검색했을 경우 공백로우값도 함꼐 나타나게 되는데


그럴땐 조건문에 아래와같은 조건을 추가해주면 해결된다.


1
 WHERE SUBSTRING_INDEX ( SUBSTRING_INDEX ( SEARCH_TAG_HISTORY . SEARCH_TEXT ,  ' ' , numbers . n ),  ' ' ,  -1 ) != ' '
cs


Mysql과 MariaDB에서 like와 in을 동시에 사용한 것과 동일한 결과를 출력하는 방법



다른 DBMS는 모르겠지만 mysql에서는 like와 in을 함께 사용 할 수가 없다. 물론 마리아DB도 마찬가지로 사용을 할 수 없다. 


 그러나 개발을 하다보면 like와 in을 동시에 사용할 수 있으면 좋겠다는 생각을 하게 되는데


그에 대한 해결책이다.


방법을 몰라 애멀게 구글에서 like in만 주구장창 검색을 하다가 겨우 찾아낸 방법이다


그리고 이렇게나 간단한 방법이 있는데도 내가 몰라서 개고생 한걸 생각하면 역시 머리가 나쁘면


손발이 고생한다는 말을 되뇌이게 된다.


매우 간단한 방법이다.



1
2
3
4
5
6
7
    SELECT  
             SEARCH_TEXT
        FROM
            search_tbl
        WHERE
               SEARCH_TEXT REGEXP  '헤이|모두들|안녕|내가|누군지|아니'
    
cs




위와 같이 조건을 주게 되면 REGEXP라는 정규식을 사용하면 


like 와 in을 동시에 사용한것과 같이 데이터 베이스의 search_tbl이라는 테이블 안에 있는  SEARCH_TEXT라는 컬럼 로우에


헤이, 모두들, 안녕, 내가, 누군지, 아니   이 6개 단어중에 하나라도 포함이 된 로우는


전부 출력을 해주게 된다.


지저분하게 검색조건에 like를 몇개씩 늘이지 않아도 된다.


저걸 like만으로 사용하게 된다면


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
        SELECT  
             SEARCH_TEXT
        FROM
            search_tbl
        WHERE
            SEARCH_TEXT like '%헤이%'
        OR
            SEARCH_TEXT like '%모두들%'
        OR
            SEARCH_TEXT like '%안녕%'
        OR
            SEARCH_TEXT like '%내가%'
        OR
            SEARCH_TEXT like '%누군지%'
        OR
            SEARCH_TEXT like '%아니%'
    
cs


이런 장황한 쿼리가 되어 버린다.



REGEXP를 사용하는 정규식에 대해선 아직 잘 모르지만 급하게 like와 in을 동시에 사용 하는 방법과


비슷한 결과를 원한다면 몰라도 원하는 값만 출력해준다면 가져다 쓰면 유용할 듯 하다.


여기에 조금더 응용을 하게 되면 REGEXP 다음에 나오는 정규식을 서브쿼리를 이용해 만들 수도 있다


예를 들어 아래와 같은 쿼리를 구현할 수가 있다


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
        SELECT  
             SEARCH_TEXT
        FROM
            search_tbl
        WHERE
               SEARCH_TEXT REGEXP (
                                        select
                                            REPLACE(GROUP_CONCAT(TAG_NAME),',','|') AS NAME
                                        from
                                            (
                                                SELECT
                                                    DISTINCT TAGLIST.TAG_NAME,
                                                    TAGLIST.ROOT_TAG
                                                FROM
                                                    ITEM_LIST LIST
                                                JOIN
                                                    ITEM_LIST TARGET
                                                ON
                                                    TARGET.ROOT_TAG = LIST.ROOT_TAG
                                                WHERE
                                                    TARGET.TAG_NAME like '조건1'
                                                OR
                                                    TARGET.TAG_NAME like '조건2'
                                                 
                                            ) selectTagList
                                        )
cs



문자열을 바꿔주는 REPLACE와 값을 이어주는 GRUOP_CONCAT 함수를 사용하며


서브쿼리로 정규식의 모양새를 만들어주고 그 결과로 조건을 만드는 방법이다.


REPLACE와 CONCAT함수는 sql 쿼리 작성시에 유용하게 자주 사용되니 알아두자


제 포스팅이 도움이 되었나요? 

그렇다면 공감하기 한번 눌러주세요 블로거에게 큰 힘이 됩니다


+ Recent posts