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

Oracle SQL - Jak uzyskać odrębną liczbę dla każdej kolumny dynamicznie?

Jeśli wszystko jest w porządku z jednym wierszem zestawu wyników na kolumnę, możesz dostosować ta magiczna sztuczka XML :

select owner, table_name, column_name,
  to_number(xmlquery('/ROWSET/ROW/C/text()'
    passing xmltype(dbms_xmlgen.getxml(
      'select count(distinct "' || column_name || '") as c '
      || 'from "' || owner || '"."' || table_name || '"'))
  returning content)) as c
from all_tab_columns
where owner = '<your table owner>'
and data_type in ('NUMBER', 'DATE', 'TIMESTAMP', 'CHAR', 'VARCHAR2', 'NCHAR', 'NVARCHAR2');

... wyszczególnienie wszystkich typów danych, których potrzebujesz, aby móc liczyć; tak naprawdę oznacza to wykluczenie te, które nie radzą sobie z distinct jak CLOB, ale ponieważ możesz mieć zagnieżdżone tabele itp., prawdopodobnie łatwiej będzie wymienić te, które robisz chcą i oczekują, że umieją liczyć.

dbms_xmlgen() wywołanie konwertuje wynik tego select count(distinct ...) ... zapytanie, które jest efektywnie konstruowane dynamicznie, do struktury XML, a następnie możesz wyciągnąć z niej licznik za pomocą XMLQuery() (zamiast przestarzałej extractvalue() w odpowiedzi, do której prowadzi link).

Jako bardzo szybkie demo:

create table t42 (id number, str varchar2(20));
insert into t42 values (1, 'Test');
insert into t42 values (2, 'Test');
insert into t42 values (3, 'Test 2');
insert into t42 values (3, null);

select owner, table_name, column_name,
  to_number(xmlquery('/ROWSET/ROW/C/text()'
    passing xmltype(dbms_xmlgen.getxml(
      'select count(distinct "' || column_name || '") as c '
      || 'from "' || owner || '"."' || table_name || '"'))
  returning content)) as c
from all_tab_columns
where owner = 'MY_SCHEMA'
and table_name = 'T42'
and data_type in ('NUMBER', 'DATE', 'TIMESTAMP', 'CHAR', 'VARCHAR2', 'NCHAR', 'NVARCHAR2');

OWNER           TABLE_NAME      COLUMN_NAME              C
--------------- --------------- --------------- ----------
MY_SCHEMA       T42             ID                       3
MY_SCHEMA       T42             STR                      2

count() funkcja ignoruje wartości null, więc aby je policzyć trzeba je przekonwertować, np. z

count(case when <your_column> is null then 1 end)

Dołącz to tutaj albo z drugą klauzulą ​​XMLQuery:

select owner, table_name, column_name,
  to_number(xmlquery('/ROWSET/ROW/C/text()'
    passing xmltype(dbms_xmlgen.getxml(
      'select count(distinct "' || column_name || '") as c '
      || 'from "' || owner || '"."' || table_name || '"'))
  returning content)) as distinct_count,
  to_number(xmlquery('/ROWSET/ROW/C/text()'
    passing xmltype(dbms_xmlgen.getxml(
      'select count(case when "' || column_name || '" is null then 1 end) as c '
      || 'from "' || owner || '"."' || table_name || '"'))
  returning content)) as null_count
from all_tab_columns
where owner = 'MY_SCHEMA'
and table_name = 'T42'
and data_type in ('NUMBER', 'DATE', 'TIMESTAMP', 'CHAR', 'VARCHAR2',
    'NCHAR', 'NVARCHAR2');

OWNER           TABLE_NAME      COLUMN_NAME     DISTINCT_COUNT NULL_COUNT
--------------- --------------- --------------- -------------- ----------
MY_SCHEMA       T42             ID                           3          0
MY_SCHEMA       T42             STR                          2          1

lub z pojedynczą tabelą XMLTable, która wyodrębnia obie wartości kolumn z wygenerowanego kodu XML, który jest modyfikowany, aby wykonywać oba zliczenia naraz:

select a.owner, a.table_name, a.column_name,
  x.distinct_count, x.null_count
from
(
  select owner, table_name, column_name,
    dbms_xmlgen.getxml(
      'select count(distinct "' || column_name || '") as c1,'
        || 'count(case when "' || column_name || '" is null then 1 end) as c2 '
      || 'from "' || owner || '"."' || table_name || '"') as xml_clob
  from all_tab_columns
  where owner = 'MY_SCHEMA'
  and table_name = 'T42'
  and data_type in ('NUMBER', 'DATE', 'TIMESTAMP', 'CHAR', 'VARCHAR2',
    'NCHAR', 'NVARCHAR2')
) a
cross join xmltable (
  '/ROWSET/ROW'
  passing xmltype(a.xml_clob)
  columns distinct_count number path 'C1',
    null_count number path 'C2'
) x;

OWNER           TABLE_NAME      COLUMN_NAME     DISTINCT_COUNT NULL_COUNT
--------------- --------------- --------------- -------------- ----------
MY_SCHEMA       T42             ID                           3          0
MY_SCHEMA       T42             STR                          2          1



  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. Czy istnieje wyrażenie CASE w JPQL?

  2. Błąd IO:Karta sieciowa nie mogła nawiązać połączenia - z Oracle 11gR2. Łączenie z programistą SQL

  3. Jak wyłączyć pulę połączeń?

  4. Odwróć w Oracle tę ścieżkę z/y/x do x/y/z

  5. Dynamiczne instrukcje Select SQL z MyBatis