Oracle
 sql >> Baza danych >  >> RDS >> Oracle

Uzyskaj dane konta w Oracle za pomocą funkcji

Wygląda na to, że opis problemu jest nieco chaotyczny, proszę sformatować pytanie i uczynić je bardziej czytelnymi. W każdym razie rozwiązania są proste. Potrzebujesz tylko jednego if oświadczenie, gdzie w zależności od poziomu przeszukujesz w pierwszej lub drugiej tabeli i odpowiedniej kolumnie:

create or replace function get_accountdetails (par_input in varchar2) return varchar2 is
  v_aid varchar2(10);
  v_db  varchar2(10);
  v_lvl varchar2(10);
  v_ret varchar2(20) := '';
begin
  v_aid := regexp_substr(par_input, '\d+', 1, 1);
  v_db  := regexp_substr(par_input, '\d+', 1, 2);
  v_lvl := regexp_substr(par_input, '\d+', 1, 3);

  if v_lvl = 1 then
     select dim_cust_key
       into v_ret
       from dim_cust_acnt
       where level1_account_id = v_aid and database_id = v_db;
  elsif v_lvl = 2 then
     select dim_cust_key
       into v_ret
       from dim_cust_dept
       where level2_account_id = v_aid and database_id = v_db;
  else
     select dim_cust_key
       into v_ret
       from dim_cust_dept
       where level3_account_id = v_aid and database_id = v_db;
  end if;
  return v_ret;
end;

Oto tabele i przykładowe wywołania funkcji:

create table dim_cust_acnt (dim_cust_key, level1_account_id, database_id) as (
    select 1123, 112, 22 from dual union all
    select 1234, 113, 23 from dual );

create table dim_cust_dept (dim_cust_key, level2_account_id, level3_account_id, database_id) as (
    select 1587, 245, 301, 21 from dual union all
    select 1576, 289, 304, 20 from dual);

select get_accountdetails('[112].[22].[1]') from dual;     -- result: 1123
select get_accountdetails('[289].[20].[2]') from dual;     -- result: 1576
select get_accountdetails('[301].[21].[3]') from dual;     -- result: 1587

Użyj odpowiednich nazw kolumn, które masz w swoich rzeczywistych danych i dostosuj typy i długość zmiennych, jeśli to konieczne. Myślę, że przydałoby się też jedno połączone zapytanie, nie potrzeba żadnej specjalnej funkcji, coś takiego jak poniżej. Użyłem full join , ponieważ Twoje przykłady nie zawierają pasujących wierszy. Prawdopodobnie proste join wystarczy.

with t(par_input) as (select '[112].[22].[1]' from dual)
select dim_cust_key
  from dim_cust_acnt a
  full join dim_cust_dept d using (dim_cust_key) 
  cross join t
  where ( 1 = regexp_substr(par_input, '\d+', 1, 3)  
          and regexp_substr(par_input, '\d+', 1, 1) = level1_account_id 
          and regexp_substr(par_input, '\d+', 1, 2) = a.database_id )
     or ( 2 = regexp_substr(par_input, '\d+', 1, 3)  
          and regexp_substr(par_input, '\d+', 1, 1) = level2_account_id 
          and regexp_substr(par_input, '\d+', 1, 2) = d.database_id )
     or ( 3 = regexp_substr(par_input, '\d+', 1, 3)  
          and regexp_substr(par_input, '\d+', 1, 1) = level3_account_id 
          and regexp_substr(par_input, '\d+', 1, 2) = d.database_id )

Wynik:

DIM_CUST_KEY
------------
        1123

Jeśli usuniesz with i cross join części i dodaj into klauzulę, możesz użyć tego zapytania w funkcji zamiast if oświadczenie.

Edytuj:

Przepraszamy za opóźnienie, ostatnio nie patrzyłem na Stack Overflow. Oto dwa przykłady pisania funkcji:

Ta funkcja zwraca połączony ciąg:

select get_details_1('[112].[22].[1],[289].[20].[2],[301].[21].[3]') as list from dual;

LIST
------------------
1123,1576,1587

Druga funkcja jest potokowa i zwraca dane jako predefiniowany zbiór ciągów, więc wartości znajdują się w osobnych wierszach.

select column_value 
  from table(get_details_2('[112].[22].[1],[289].[20].[2],[301].[21].[3]'));

COLUMN_VALUE
------------
        1123
        1576
        1587

Możesz także najpierw przeanalizować wszystkie dane wejściowe, zapisać je w jakiejś kolekcji, a następnie użyć zbierania zbiorczego w jednym zapytaniu. Jest wiele rozwiązań i możliwości, osobiście użyłbym funkcji potokowej, ale zależy to od tego, jakiej formy wyjścia potrzebujesz (zbiór czy połączony ciąg). Możesz także dodać begin ... end zablokuj i obsługuj wyjątek when no_data_found . Możesz wtedy podać specjalne informacje lub przerwać wykonanie, w zależności od tego, jakiego zachowania oczekujemy w takiej sytuacji.

Funkcja 1:

create or replace function get_details_1 (par_input in varchar2) return varchar2 is
    v_aid varchar2(10);
    v_db  varchar2(10);
    v_lvl varchar2(10);
    v_ret varchar2(20);
    v_all varchar2(200) := '';

    i_cnt int := 0;
begin
    loop
        v_aid := regexp_substr(par_input, '\d+', 1, i_cnt + 1);
        v_db  := regexp_substr(par_input, '\d+', 1, i_cnt + 2);
        v_lvl := regexp_substr(par_input, '\d+', 1, i_cnt + 3);
        i_cnt := i_cnt + 3;
    exit when v_aid is null;
        select dim_cust_key
          into v_ret
          from dim_cust_acnt a
          full join dim_cust_dept d using (dim_cust_key)
          where (v_lvl = 1 and level1_account_id = v_aid and a.database_id = v_db)
             or (v_lvl = 2 and level2_account_id = v_aid and d.database_id = v_db)
             or (v_lvl = 3 and level3_account_id = v_aid and d.database_id = v_db);
       v_all := v_all||','||v_ret;
  end loop;
  return ltrim(v_all, ',');
end;

Funkcja 2:

create or replace function get_details_2 (par_input in varchar2) 
    return sys.odcinumberlist pipelined is

    v_aid varchar2(10);
    v_db  varchar2(10);
    v_lvl varchar2(10);
    v_ret varchar2(20);
    i_cnt int := 0;
begin
    loop
        v_aid := regexp_substr(par_input, '\d+', 1, i_cnt + 1);
        v_db  := regexp_substr(par_input, '\d+', 1, i_cnt + 2);
        v_lvl := regexp_substr(par_input, '\d+', 1, i_cnt + 3);
        i_cnt := i_cnt + 3;
    exit when v_aid is null;
        select dim_cust_key
          into v_ret
          from dim_cust_acnt a
          full join dim_cust_dept d using (dim_cust_key)
          where (v_lvl = 1 and level1_account_id = v_aid and a.database_id = v_db)
             or (v_lvl = 2 and level2_account_id = v_aid and d.database_id = v_db)
             or (v_lvl = 3 and level3_account_id = v_aid and d.database_id = v_db);
       pipe row (v_ret);
  end loop;
  return;
end;



  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. problem ze znalezieniem listy plików w katalogu

  2. Przykład zbiorczego zbierania danych Oracle PL/SQL z wyjątkami zapisu

  3. Jak pogrupować według miesiąca, w tym wszystkie miesiące?

  4. Zrozumienie aliasów Oracle — dlaczego alias nie jest rozpoznawany w zapytaniu, chyba że jest umieszczony w drugim zapytaniu?

  5. Deklarowanie zmiennej tymczasowej w bazie danych Oracle