2012-11-02 21 views
6

क्या ऑरैकल में listunagg फ़ंक्शन की तरह ऐसी चीज है? उदाहरण के लिए, अगर मैं की तरह एक डेटा है: मैं इस तरह डेटा दिखाना चाहते हैंlistunagg फ़ंक्शन?

------------------------------------------------------------ 
| user_id | degree_fi | degree_en  | degree_sv  | 
-------------------------------------------------------------- 
| 3601464 | 3700   | 1600   | 2200   | 
| 1020 | 100   | 0    | 0    | 
| 3600520 | 100,3200,400 | 1300, 800, 3000 | 1400, 600, 1500 | 
| 3600882 | 0   | 100   | 200   | 
-------------------------------------------------------------- 

और:

----------------------------------------------- 
| user_id | degree_fi | degree_en | degree_sv | 
----------------------------------------------- 
| 3601464 | 3700  | 1600  | 2200  | 
| 1020 | 100  | 0  | 0  | 
| 3600520 | 100  | 1300  | 1400  | 
| 3600882 | 0  | 100  | 200  | 
| 3600520 | 3200  | 800  | 600  | 
| 3600520 | 400  | 3000  | 1500  | 
----------------------------------------------- 

मैं listagg के विपरीत जैसे कुछ समारोह को खोजने की कोशिश की लेकिन नहीं पा सके कोई। अग्रिम धन्यवाद :-)

+0

नहीं, ऐसा कोई मूल कार्य नहीं है। केवल कस्टम वर्कअराउंड –

+1

संभावित डुप्लिकेट [सीएसवी को ऑरैकल में टेबल में कैसे परिवर्तित करें] (http://stackoverflow.com/questions/3142665/how-to-convert-csv-to-table-in-oracle) – APC

उत्तर

7

जैसा कि @be अब यहां टिप्पणी में नोट किया गया है ओरेकल ऐसा कार्य प्रदान नहीं करता है।

with t1(user_id, degree_fi, degree_en, degree_sv) as 
(
    select 3601464, '3700', '1600', '2200' from dual union all 
    select 1020 , '100' , '0' , '0' from dual union all 
    select 3600520, '100,3200,400', '1300, 800, 3000', '1400, 600, 1500' from dual union all 
    select 3600882, '0', '100', '200' from dual 
), 
Occurence(ocr) as(
    select Level as ocr 
    from (select max(greatest(regexp_count(degree_fi, '[^,]+') 
          , regexp_count(degree_en, '[^,]+') 
          , regexp_count(degree_sv, '[^,]+') 
          ) 
        ) mx 
      from t1  
     ) 
    connect by level <= mx 
) 
select * 
    from (
select User_id 
    , regexp_substr(degree_fi, '[^,]+', 1, o.ocr) as degree_fi 
    , regexp_substr(degree_en, '[^,]+', 1, o.ocr) as degree_en 
    , regexp_substr(degree_sv, '[^,]+', 1, o.ocr) as degree_sv 
    from t1 t 
    cross join Occurence o 
) 
where degree_fi is not null 
    or degree_en is not null 
    or degree_sv is not null 

परिणाम::

User_Id Degree_Fi Degree_En Degree_Sv 
------------------------------------------------------------ 
3601464 3700  1600  2200 
1020  100  0   0 
3600520 100  1300  1400 
3600882 0   100  200 
3600520 3200  800  600 
3600520 400  3000  1500 
+1

धन्यवाद यह है मुझे क्या चाहिए :-) – Jaanna

0

पर विचार टॉम क्या Oracle की पर कहने के लिए "टॉम पूछें" देखना है एक सूची unagg करने के लिए http://www.oracle.com/technetwork/issue-archive/2007/07-mar/o27asktom-084983.html कोड लिस्टिंग 3 तो एक त्वरित समाधान के रूप में आप समान क्वेरी लिख सकता है या 4.

मेरा पसंदीदा विकल्प जो टॉम चर्चा नहीं करता है, छोटे तारों के लिए अच्छा है (< 34 वर्ण)। मैं ओरेकल डीबीएमएस_UTILITY.comma_to_table फ़ंक्शन का उपयोग करता हूं। उदाहरण:

SET SERVEROUTPUT ON 
DECLARE 
/** test data **/ 
    L_LIST1 VARCHAR2(500) := '"A","B","C","Pierre - Andre","D","E","OFVampFVapos;CBryan","F","G","H","I","J"'; 
    l_list2 VARCHAR2(500); 
    l_tablen BINARY_INTEGER; 
    l_tab  DBMS_UTILITY.uncl_array; 
BEGIN 
    DBMS_OUTPUT.put_line('l_list1 : ' || l_list1); 

    DBMS_UTILITY.comma_to_table (
    list => l_list1, 
    tablen => l_tablen, 
    tab => l_tab); 

    FOR i IN 1 .. l_tablen LOOP 
    DBMS_OUTPUT.put_line(i || ' : ' || l_tab(i)); 
    END LOOP; 

    DBMS_UTILITY.table_to_comma (
    tab => l_tab, 
    tablen => l_tablen, 
    list => l_list2); 

    DBMS_OUTPUT.put_line('l_list2 : ' || l_list2); 
end;