Oracle 多行记录合并/连接/聚合字符串的几种方法
怎么合并多行記錄的字符串,一直是oracle新手喜歡問的SQL問題之一,關(guān)于這個(gè)問題的帖子我看過不下30個(gè)了,現(xiàn)在就對(duì)這個(gè)問題,進(jìn)行一個(gè)總結(jié)。
-
什么是合并多行字符串(連接字符串)呢,例如:
?
要求得到如下結(jié)果集:
1 ------- -------------------- 2 中國 臺(tái)北,香港,上海 3 日本 東京,大阪?
實(shí)際就是對(duì)字符實(shí)現(xiàn)一個(gè)聚合功能,我很奇怪為什么Oracle沒有提供官方的聚合函數(shù)來實(shí)現(xiàn)它呢:)
下面就對(duì)幾種經(jīng)常提及的解決方案進(jìn)行分析(有一個(gè)評(píng)測(cè)標(biāo)準(zhǔn)最高★★★★★):
1.被集合字段范圍小且固定型 靈活性★ 性能★★★★ 難度 ★
這種方法的原理在于你已經(jīng)知道CITY字段的值有幾種,且還不算太多,如果太多這個(gè)SQL就會(huì)相當(dāng)?shù)拈L(zhǎng)。。看例子:
?
大家一看,估計(jì)就明白了(如果不明白,好好補(bǔ)習(xí)MAX DECODE和分組)。這種方法無愧為最笨的方法,但是對(duì)某些應(yīng)用來說,最有效的方法也許就是它。
2.固定表固定字段函數(shù)法 靈活性★★ 性能★★★★ 難度 ★★
此法必須預(yù)先知道是哪個(gè)表,也就是說一個(gè)表就得寫一個(gè)函數(shù),不過方法1的一個(gè)取值就要便捷多了。在大多數(shù)應(yīng)用中,也不會(huì)存在大量這種合并字符串的需求。廢話完畢,看下面:
定義一個(gè)函數(shù)
?
使用:
1 SQL> select DISTINCT(T.country),list_func1(t.country) from test t; 2 3 COUNTRY LIST_FUNC1(T.COUNTRY) 4 -------------------- ---------------- 5 中國 臺(tái)北, 香港, 上海 6 日本 東京, 大阪 7 8 SQL> select t.country,str_list(t.country) from test t GROUP BY t.country; 9 10 11 COUNTRY STR_LIST(T.COUNTRY) 12 -------------------- ----------------------- 13 中國 臺(tái)北, 香港, 上海 14 日本 東京, 大阪?
這個(gè)時(shí)候,使用分組和求唯一都可以滿足要求。它的原理就是,根據(jù)唯一的分組字段country,在函數(shù)里面再次查詢?cè)撟侄螌?duì)應(yīng)的所有被合并列,使用PL/SQL將其合并輸出。
3.靈活表函數(shù)法 靈活性★★★ 性能★★★ 難度 ★★★
該方法是在方法2的基礎(chǔ)上,使用動(dòng)態(tài)SQL,將表名和字段名稱傳入,從而達(dá)到靈活的目的。
?
4.一條SQL法 靈活性★★★★ 性能★★ 難度 ★★★★
一條SQL的法則是某位大師提出的,大家曾經(jīng)在某個(gè)時(shí)期都樂此不彼的尋求各種的問題一條SQL法,但是大師的意思似乎被曲解,很多性能差,可讀性差,靈活差的SQL都是這個(gè)原則產(chǎn)物,所謂畫虎不成反成犬類。不過,解決問題始終是第一原則,這里還是給出一個(gè)比較有代表性的一條SQL方法。
?
step 2 創(chuàng)造子節(jié)點(diǎn)父節(jié)點(diǎn)
1 SQL> SELECT country,city,country||rn rchild,country||(rn-1) rfather 2 2 FROM 3 3 (SELECT test.country ,test.city,row_number() over (PARTITION BY test.country ORDER BY test.city) rn 4 4 FROM test) 5 5 / 6 日本 大阪 日本1 日本0 7 日本 東京 日本2 日本1 8 中國 上海 中國1 中國0 9 中國 臺(tái)北 中國2 中國1 10 中國 香港 中國3 中國2 11 step 3 利用sys_connect_by_path生成結(jié)果集 12 SELECT country,sys_connect_by_path(city,',') city 13 FROM 14 (SELECT country,city,country||rn rchild,country||(rn-1) rfather 15 FROM 16 (SELECT test.country ,test.city,row_number() over (PARTITION BY test.country ORDER BY test.city) rn FROM test)) CONNECT BY PRIOR rchild=rfather START WITH rfather LIKE '%0' 17 日本 ,大阪 18 日本 ,大阪,東京 19 中國 ,上海 20 中國 ,上海,臺(tái)北 21 中國 ,上海,臺(tái)北,香港 22 step 4 最終步驟,篩選結(jié)果集合 23 SQL> SELECT country,max(substr(city,2)) city 24 2 FROM 25 3 (SELECT country,sys_connect_by_path(city,',') city 26 4 FROM 27 5 (SELECT country,city,country||rn rchild,country||(rn-1) rfather 28 6 FROM 29 7 (SELECT test.country ,test.city,row_number() over (PARTITION BY test.country ORDER BY test.city) rn 30 8 FROM test)) 31 9 CONNECT BY PRIOR rchild=rfather START WITH rfather LIKE '%0') 32 10 GROUP BY country; 33 34 35 COUNTRY CITY 36 -------------------- ------- 37 中國 上海,臺(tái)北,香港 38 日本 大阪,東京?
可謂是,7歪8搞,最后還是弄出來了,呵呵。 PS:(邏輯上是對(duì)的..但是寫的比較繁瑣,可以簡(jiǎn)化!)
5.自定義聚合函數(shù) 靈活性★★★★★ 性能★★★★★ 難度 ★★★★★
最后一個(gè)方法是我認(rèn)為“王道”的方法,自定義聚合函數(shù)。
就如何我在本開始說的,為啥oracle沒有這種聚合函數(shù)呢?我也不知道,但oracle提供了聚合函數(shù)的API可以讓我方便的自己定義聚合函數(shù)。
詳細(xì)可以看Oracle Data Catridge guide這個(gè)文檔。連接如下:
http://www.oracle.com.cn/other/9ionlinedoc/appdev.920/a96595/toc.htm
下面給出一個(gè)簡(jiǎn)單的例子:
?
6.待發(fā)掘...
PS: 在 oracle 10g下,可以使用以下系統(tǒng)函數(shù):
1 select id,WMSYS.WM_CONCAT(oid) oid 2 from table1 3 group by id?
總結(jié),合并字符串還有更多的方法希望大家能發(fā)掘,本文的目的主要是拋磚引玉,如果有新的發(fā)現(xiàn)我會(huì)繼續(xù)更新方法。
需要注意的問題是,本文采用varchar2為例子,所以長(zhǎng)度有限制,oracle的版本對(duì)方法的實(shí)現(xiàn)也影響。
轉(zhuǎn)載于:https://www.cnblogs.com/zhangwei595806165/p/3142446.html
總結(jié)
以上是生活随笔為你收集整理的Oracle 多行记录合并/连接/聚合字符串的几种方法的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Ribbon_窗体_实现Ribbon风格
- 下一篇: .configure