sexta-feira, 8 de abril de 2011

Tabelas externas no Oracle


“Tabelas externas” são objetos que permitem consultar informações em um arquivo sob o formato de texto, como se o arquivo fosse uma tabela do banco de dados. Esse dispositivo foi criando a partir da versão 9i, as operações permitidas eram somente de leitura, mas a partir da versão 10g, também foi incluído também a possibilidade para gerar informações para uma tabela externa.

A definição de tabela externa assemelha-se a uma “view” que permite o banco de dados consultar os dados externos sem que os dados estejam carregados na base de dados. Pode-se, por exemplo realizar operações de “selects”, “joins”, “order by” usando dados das tabelas do banco de dados e dados externos. Pode-se ainda criar views e sinônimos para tabelas externas.

O banco de dados Oracle fornece os meios de definição dos metadados para tabelas externas através do comando “CREATE TABLE... ORGANIZATION EXTERNAL”.


Neste artigo serão descritos os passos para a criação e manutenção de uma tabela externa !

Para demonstrar a criação e manutenção de uma tabela externa criaremos um arquivo texto chamado “estados.dat”, cujo o conteúdo segue a abaixo.

SP,Sao Paulo
RJ,Rio de Janeiro
MG,Minas Gerais
TO,Tocantins
MA,Maranhao
AM,Amazonas
RS,Rio Grande do Sul
MS,Mato Grosso do Sul
SC,Santa Catarina
PR,Parana
DF,Distrito Federal
PE,Pernambuco

[oracle]$ pwd
/home/oracle/dir

[oracle]$ ls -ltr
-rw-r--r--  1 oracle dba 190 Mar 14 17:21 estados.dat

Este arquivo foi criado no caminho [/home/oracle/dir] e deverá ser referenciado no banco de dados através da criação de um diretório.

Abaixo temos a criação de um diretório dentro do banco de dados, para indicar onde o arquivo texto “estados.dat” está localizado.

SQL> CREATE OR REPLACE DIRECTORY EXT_TABLES AS '/home/oracle/dir';

Directory created.


Após a criação do diretório deve-se criar a “tabela externa” para acessar o arquivo texto.

SQL> CREATE TABLE ESTADOS_EXT
      (
       SIGLA VARCHAR2(2),
       NOME_ESTADO VARCHAR2(40)
       )
      ORGANIZATION EXTERNAL
       (
         TYPE ORACLE_LOADER
         DEFAULT DIRECTORY EXT_TABLES
         ACCESS PARAMETERS
         (
          RECORDS DELIMITED BY NEWLINE
          FIELDS TERMINATED BY ','
          MISSING FIELD VALUES ARE NULL
         )
         LOCATION ('estados.dat')
       )
     REJECT LIMIT UNLIMITED;


Table created.

A cláusula REJECT LIMIT especifica que não existirão limites para o número de erros que poderão ocorrer durante uma consulta ao dado externo. (mais informações ver manual Oracle 11g SQL Reference).

Para consultar as informações, basta simplesmente executar o “select” abaixo.

SQL> SELECT * FROM ESTADOS_EXT;

SI NOME_ESTADO
-- ----------------------------------------
SP Sao Paulo
RJ Rio de Janeiro
MG Minas Gerais
TO Tocantins
MA Maranhao
AM Amazonas
RS Rio Grande do Sul
MS Mato Grosso do Sul
SC Santa Catarina
PR Parana
DF Distrito Federal
PE Pernambuco


13 rows selected.


Para criar uma view a partir de uma “tabela externa”, usar:

SQL> CREATE VIEW ESTADOS_TESTE
AS SELECT * FROM  ESTADOS_EXT
    WHERE SIGLA LIKE 'M%';

View created.


Acessando dados da view ESTADOS_TESTE.


SQL> SELECT * FROM ESTADOS_TESTE;

SI NOME_ESTADO
-- ----------------------------------------
MG Minas Gerais
MA Maranhao
MS Mato Grosso do Sul

As informações dos objetos podem ser acessadas através das “views”  DBA_OBJECTS, ALL_OBJECTS ou USER_OBJECTS.

SQL> SELECT OBJECT_NAME, OBJECT_TYPE FROM ALL_OBJECTS
     WHERE OBJECT_NAME LIKE 'ESTADOS_EXT';

OBJECT_NAME                    OBJECT_TYPE
------------------------------ ------------------
ESTADOS_EXT                    TABLE


SQL> SELECT OBJECT_NAME, OBJECT_TYPE FROM ALL_OBJECTS
     WHERE OBJECT_NAME LIKE 'EXT_TABLES';

OBJECT_NAME                    OBJECT_TYPE
------------------------------ ------------------
EXT_TABLES                     DIRECTORY



Incluindo uma coluna numa tabela externa

Inicialmente no arquivo texto chamado “estados.dat”, foi incluída uma nova coluna de regiões.

SP,Sao Paulo,Sudeste
RJ,Rio de Janeiro,Sudeste
MG,Minas Gerais,Sudeste
TO,Tocantins,Centro-Oeste
MA,Maranhao,Centro-Oeste
AM,Amazonas,Norte
RS,Rio Grande do Sul,Sul
MS,Mato Grosso do Sul,Centro-Oeste
SC,Santa Catarina,Sul
PR,Parana,Sul
DF,Distrito Federal,Centro-Oeste
PE,Pernambuco,Nordeste


Para alterar a tabela externa no banco de dados, utilizar o comando “ALTER TABLE”.

SQL>  ALTER TABLE ESTADOS_EXT ADD(REGIAO VARCHAR(40));

Table altered.

SQL> DESC ESTADOS_EXT

 Name                                      Null?    Type
 ----------------------------------------- -------- ---------------------------
 SIGLA                                              VARCHAR2(2)
 NOME_ESTADO                                        VARCHAR2(40)
 REGIAO                                             VARCHAR2(40)


Consultando as informações.

SQL> SELECT * FROM ESTADOS_EXT;

SI NOME_ESTADO                              REGIAO
-- --------------------------------------- ------------------------------------
SP Sao Paulo                                Sudeste
RJ Rio de Janeiro                           Sudeste
MG Minas Gerais                             Sudeste
TO Tocantins                                Centro-Oeste
MA Maranhao                                 Centro-Oeste
AM Amazonas                                 Norte
RS Rio Grande do Sul                        Sul
MS Mato Grosso do Sul                       Centro-Oeste
SC SantaCatarina                            Sul
PR Parana                                   Sul
DF Distrito Federal                         Centro-Oeste
PE Pernambuco                               Nordeste


13 rows selected.


Carregando informações de texto para tabelas do banco de dados

Para carregar informações de texto para as tabelas do banco de dados, a partir das “tabelas externas”, deve-se utilizar o comando "INSERT INTO… SELECT FROM" ao invés de utilizar o utilitário SQL*Loader. Este método é mais prático e eficiente.

Criando uma tabela vazia.

SQL> CREATE TABLE ESTADOS (
     UF CHAR(2),
     NOME VARCHAR2(40));

Table created.


SQL> DESC ESTADOS
 Name                                      Null?    Type
 ----------------------------------------- -------- ---------------------------
 UF                                                 CHAR(2)
 NOME                                               VARCHAR2(40)


Carregando informações para a tabela no banco de dados.

SQL> INSERT INTO ESTADOS (UF,NOME) SELECT * FROM ESTADOS_EXT;

13 rows created.

SQL> COMMIT;

Commit complete.


Acessando dados na tabela.

SQL> SELECT * FROM ESTADOS;

UF NOME
-- ----------------------------------------
SP Sao Paulo
RJ Rio de Janeiro
MG Minas Gerais
TO Tocantins
MA Maranhao
AM Amazonas
RS Rio Grande do Sul
MS Mato Grosso do Sul
SC Santa Catarina
PR Parana
DF Distrito Federal
PE Pernambuco


13 rows selected.


Criando uma “tabela externa” através de tabelas do banco de dados

Para criar uma “tabela externa” a partir de uma tabela do banco de dados, basta utilizar o comando “CREATE TABLE...ORGANIZATION EXTERNAL” (opção disponível a partir da versão 10g).

Consulta a tabela do banco de dados “estados”.

SQL> DESC ESTADOS

 Name                                      Null?    Type
 ---------------------------------------- -------- ----------------------------
 UF                                                 CHAR(2)
 NOME                                               VARCHAR2(40)


SQL> SELECT * FROM ESTADOS;

UF NOME
-- ----------------------------------------
SP Sao Paulo
RJ Rio de Janeiro
MG Minas Gerais
TO Tocantins
MA Maranhao
AM Amazonas
RS Rio Grande do Sul
MS Mato Grosso do Sul
SC Santa Catarina
PR Parana
DF Distrito Federal
PE Pernambuco
BA Bahia
ES Espirito Santo
SE Sergipe
RN Rio Grande do Norte
AL Alagoas
CE Ceara
PB Paraiba
RR Roraima
RO Rondonia
MT Mato Grosso
AC Acre
PA Para
PI Piaui
GO Goias

26 rows selected.


Criação da “tabela externa” export_estados.

SQL>CREATE TABLE EXPORT_ESTADOS
      ORGANIZATION EXTERNAL
      (TYPE ORACLE_DATAPUMP
       DEFAULT DIRECTORY EXT_TABLES
       LOCATION ('estados.txt')
      ) AS SELECT * FROM ESTADOS;


Table created.


Verificação da nova tabela externa criada.


SQL> SELECT * FROM EXPORT_ESTADOS;

UF NOME
-- ----------------------------------------
SP Sao Paulo
RJ Rio de Janeiro
MG Minas Gerais
TO Tocantins
MA Maranhao
AM Amazonas
RS Rio Grande do Sul
MS Mato Grosso do Sul
SC Santa Catarina
PR Parana
DF Distrito Federal
PE Pernambuco
BA Bahia
ES Espirito Santo
SE Sergipe
RN Rio Grande do Norte
AL Alagoas
CE Ceara
PB Paraiba
RR Roraima
RO Rondonia
MT Mato Grosso
AC Acre
PA Para
PI Piaui
GO Goias

26 rows selected.



Lendo informações de arquivos textos compactados

A partir da versão 11g, o Oracle permite que os arquivos externos possam estar compactados (nos formatos *.zip, *.gzip,  entre outros), para utilizar essa funcionalidade deve-se seguir os passos abaixo:

Criar um diretório onde serão executadas as descompactações.
SQL> CREATE OR REPLACE DIRECTORY EXECDIR AS '/home/oracle/dir';
Directory created.


Dar permissão de execução no diretório criado.

SQL> GRANT EXECUTE    ON DIRECTORY EXECDIR    TO PUBLIC;

Grant succeeded.

 

 

Dar permissão de leitura e gravação no diretório onde ficará armazenado o arquivo compactado.

SQL> GRANT READ, WRITE ON DIRECTORY EXT_TABLES TO PUBLIC;

Grant succeeded.

 

 

Criar a nova tabela externa.

 

SQL> CREATE TABLE ESTADOS_EXT_ZIP

  2        (

  3         SIGLA       VARCHAR2(2),

  4         NOME_ESTADO VARCHAR2(40)

  5         )

  6        ORGANIZATION EXTERNAL

  7         (

  8           TYPE ORACLE_LOADER

  9           DEFAULT DIRECTORY EXT_TABLES

 10           ACCESS PARAMETERS

 11           (

 12            RECORDS DELIMITED BY NEWLINE

 13            PREPROCESSOR EXECDIR:'zcat' -- > Parâmetro necessário para

 14                                             executar a descompactação.

 15            FIELDS TERMINATED BY ","

 16           )

 17           LOCATION ('estBR.zip') -- > Arquivo compactado.

 18         )

 19       REJECT LIMIT UNLIMITED;

 

 

Table created.

 


Para acessar os dados da tabela basta somente executar um “select”.

 

SQL> select * from ESTADOS_EXT_ZIP;

 

SI NOME_ESTADO

-- ----------------------------------------

SP Sao Paulo

RJ Rio de Janeiro

MG Minas Gerais

TO Tocantins

MA Maranhao

AM Amazonas

RS Rio Grande do Sul

MS Mato Grosso do Sul

SC Santa Catarina

PR Parana

DF Distrito Federal

PE Pernambuco

BA Bahia

ES Espirito Santo

SE Sergipe

RN Rio Grande do Norte

AL Alagoas

CE Ceara

PB Paraiba

RR Roraima

RO Rondonia

MT Mato Grosso

AC Acre

PA Para

PI Piaui

GO Goias

 

26 rows selected.

 

 

Arquivos de apoio utilizados para esta funcionalidade:

Conteúdo do arquivo “estBR.zip” este deverá ser compactado e gravado no mesmo diretório indicado comando “create directory” no banco de dados, para esse exemplo será no caminho [/home/oracle/dir].

SP,Sao Paulo

RJ,Rio de Janeiro

MG,Minas Gerais

TO,Tocantins

MA,Maranhao

AM,Amazonas

RS,Rio Grande do Sul

MS,Mato Grosso do Sul

SC,Santa Catarina

PR,Parana

DF,Distrito Federal

PE,Pernambuco

BA,Bahia

ES,Espirito Santo

SE,Sergipe

RN,Rio Grande do Norte

AL,Alagoas

CE,Ceara

PB,Paraiba

RR,Roraima

RO,Rondonia

MT,Mato Grosso

AC,Acre

PA,Para

PI,Piaui

GO,Goias


Conteúdo do script shell "zcat", utilizado para a descompactação dinâmica do arquivo.

/bin/zcat -f $1

 

Onde a opção “-f” irá forçar a descompactação mesmo que o arquivo já exista e a variável “$1” servirá de passagem para o comando “zcat”.

 

Removendo tabelas externas

 

Para remover uma tabela externa, basta executar o comando “DROP TABLE”, este comando irá remover o metadado do banco de dados, não afetando os dados que estão gravados externamente em disco.

Restrições de tabelas externas



  • Tabelas externas não suportam colunas criptografadas, tanto para exportação quanto para importação. 
  • Não descrevem todos os dados que são armazenados no banco de dados. 
  • Não descrevem como os dados são armazenados na fonte externa. Esta é a função dos parâmetros de acesso. 
  • Para processamento de colunas: O recurso de tabelas externas recupera todas as colunas definidas para uma tabela externa. Isso garante um resultado consistente conjunto de todas as consultas. No entanto, por motivos de desempenho, pode-se optar em processar apenas as colunas referenciadas de uma tabela externa, minimizando a quantidade de conversão e manipulação de dados necessária para executar uma consulta. 
  • Não carrega dados de colunas do tipo LONG. 
  • Strings de SQL não podem ser especificados nos parâmetros de acesso ao driver ORACLE_LOADER. Como solução, pode-se usar a cláusula DECODE no comando SELECT que está lendo a tabela externa. Outra opção seria criar uma “view” da tabela externa que usa a cláusula DECODE e selecionar os dados a partir dessa “view”, dispensando o acesso a tabela externa. 
  • Os identificadores (por exemplo, nomes de colunas ou nome de tabelas) são especificados nos parâmetros de acesso externo da tabela, certos valores são considerados palavras reservadas pelo banco de dados. Se uma palavra reservada for utilizada como um identificador, ele deverá ser colocado entre aspas duplas “”.


    Conclusão

     


    Através deste artigo vimos algumas possibilidades de trabalho com “tabelas externas”, que permitem o acesso a informações ainda em modo texto. Tabelas externas fornecem uma maneira conveniente e prática para manipular dados dentro e fora do banco de dados. Na próxima vez que um processo de carga for definido em seu ambiente considere o uso tabelas externas.

    Maiores informações podem ser obtidas através do manual (Oracle Database Utilities 11g).




    Rubens Oliveira
    DBA Oracle Consultor
    olivert.dba@consultant.com