2011-01-27 14 views
6

PL/SQL koleksiyonundaki bir elemanın (ilk) indeksini belirlemek için dahili bir fonksiyon var mı?PL/SQL koleksiyonunda öğe endeksinin alınması

şey

A: 1 
B: 2 
C: 

DECLARE 
    TYPE t_test IS TABLE OF VARCHAR2(1); 
    v_test t_test; 
BEGIN 
    v_test := NEW t_test('A', 'B', 'A'); 
    dbms_output.put_line('A: ' || get_index(v_test, 'A')); 
    dbms_output.put_line('B: ' || get_index(v_test, 'B')); 
    dbms_output.put_line('C: ' || get_index(v_test, 'C')); 
END; 

gibi gerekli her türlü Birleştirici Diziler, İç İçe Tablolar veya Varrays kullanabilirsiniz. Aynı öğe birden fazla kez varsa, ilk oluşumun indeksi yeterlidir.


Aksi takdirde belgelere bakın Şüpheniz

CREATE FUNCTION get_index (in_test IN t_test, in_value IN VARCHAR2) 
    RETURN PLS_INTEGER 
AS 
    i PLS_INTEGER; 
BEGIN 

i := in_test.FIRST; 
    WHILE(i IS NOT NULL) LOOP 
    IF(in_test(i) = in_value) THEN 
     RETURN i; 
    END IF; 

    i := in_test.NEXT(i); 
    END LOOP; 

    RETURN NULL; 

END get_index; 

cevap

0

zaman böyle bir şey yapmak zorunda ederim;)() here

DECLARE 
    TYPE aa_type_int IS TABLE OF INTEGER INDEX BY PLS_INTEGER; 
    aa_int aa_type_int; 

    PROCEDURE print_first_and_last IS 
    BEGIN 
    DBMS_OUTPUT.PUT_LINE('FIRST = ' || aa_int.FIRST); 
    DBMS_OUTPUT.PUT_LINE('LAST = ' || aa_int.LAST); 
    END print_first_and_last; 

BEGIN 
    aa_int(1) := 3; 
    aa_int(2) := 6; 
    aa_int(3) := 9; 
    aa_int(4) := 12; 

    DBMS_OUTPUT.PUT_LINE('Before deletions:'); 
    print_first_and_last; 

    aa_int.DELETE(1); 
    aa_int.DELETE(4); 

    DBMS_OUTPUT.PUT_LINE('After deletions:'); 
    print_first_and_last; 
END; 
/

Sonuç:

Before deletions: 
FIRST = 1 
LAST = 4 
After deletions: 
FIRST = 2 
LAST = 3 
+4

Bu, belirli bir değere sahip bir öğenin ilk dizini değil, tüm öğelerin ilk ve son dizinini veriyor. OP, örneğinizde, örneğin, silmeden önce veya sonra indeks numarası 3'ü döndürmesi gereken 9'un ilk görüntüsüne karşılık gelen endeksi nasıl bulacağımızı bilmek ister. –

+0

@AlexPoole: Teşekkürler, hiçbir şey kalmadı :) –

6

Elbette, eğer bu Gerçekten yardımcı olur, yoksa düşünüyorsanız daha zariftir: Bir koleksiyonu arar yerleşik bir işlevi olduğunu sanmıyorum

create type t_test as table of varchar2(1); 
/

DECLARE 
--TYPE t_test IS TABLE OF VARCHAR2(1); 
    v_test t_test; 

    function get_index(q in t_test, c in varchar2) return number is 
    ind number; 
    begin 
    select min(rn) into ind from (
     select column_value cv, rownum rn 
     from table(q) 
    ) 
    where cv = c; 

    return ind; 
    end get_index; 

BEGIN 
    v_test := NEW t_test('A', 'B', 'A'); 

    dbms_output.put_line('A: ' || get_index(v_test, 'A')); 
    dbms_output.put_line('B: ' || get_index(v_test, 'B')); 
    dbms_output.put_line('C: ' || get_index(v_test, 'C')); 
END; 
/

show errors 

drop type t_test; 
+0

Teşekkürler René. Yerleşik bir şey yok gibi görünüyor, bu yüzden çözümünüzü kullanabilirim. '+ 1' ama henüz kabul etmiyorum, çünkü bu soruya gerçekten cevap vermiyor :) –

2

. Ancak, bir koleksiyonda çok fazla arama yapmanız gerektiğini biliyorsanız, bir dizin oluşturabilirsiniz. Koleksiyona eleman eklemek biraz daha pahalı olacaktır, ancak bir eleman aramak bir O (1) işlem olacaktır (kaba kuvvet arama için O (n) yerine). Örneğin, böyle bir şey kullanabilirsiniz:

SQL> DECLARE 
    2  TYPE t_test IS TABLE OF VARCHAR2(1); 
    3  TYPE t_test_r IS TABLE OF NUMBER INDEX BY VARCHAR2(1); 
    4 
    5  v_test t_test; 
    6  v_test_r t_test_r; 
    7 
    8  FUNCTION get_index(p_test_r t_test_r, 
    9      p_element VARCHAR2) RETURN NUMBER IS 
10  BEGIN 
11  RETURN p_test_r(p_element); 
12  EXCEPTION 
13  WHEN no_data_found THEN 
14   RETURN NULL; 
15  END get_index; 
16 
17  PROCEDURE add_element(p_test IN OUT t_test, 
18       p_test_r IN OUT t_test_r, 
19       p_element VARCHAR2) IS 
20  BEGIN 
21  p_test.extend; 
22  p_test(p_test.count) := p_element; 
23  p_test_r(p_element) := least(p_test.count, 
24          nvl(get_index(p_test_r, p_element), 
25           p_test.count)); 
26  END add_element; 
27 BEGIN 
28  v_test := NEW t_test(); 
29  add_element(v_test, v_test_r, 'A'); 
30  add_element(v_test, v_test_r, 'B'); 
31  add_element(v_test, v_test_r, 'A'); 
32  dbms_output.put_line('A: ' || get_index(v_test_r, 'A')); 
33  dbms_output.put_line('B: ' || get_index(v_test_r, 'B')); 
34  dbms_output.put_line('C: ' || get_index(v_test_r, 'C')); 
35 END; 
36/

A: 1 
B: 2 
C: 

PL/SQL procedure successfully completed 

Ayrıca diziler ve diziler bu kayıt türünü kullanmak istiyorsunuz ile etkileşim tüm işlevleri/prosedürleri hem içeren bir kayıt tanımlayabiliriz.