Criar e gerenciar relacionamentos de chave estrangeira

Nesta página, descrevemos como gerenciar relações de chave externa no banco de dados.

Uma chave externa é uma coluna compartilhada entre tabelas para estabelecer um link entre dados relacionados. Quando você usa uma chave estrangeira, o Spanner garante que essa relação seja mantida.

O diagrama a seguir mostra um esquema de banco de dados básico em que os dados de uma tabela têm uma relação com os dados de outra.

Exemplo de esquema de banco de dados mostrando relações de chave externa entre tabelas.

Figura 1. Diagrama de um esquema de banco de dados de processamento de pedidos

Há três tabelas no esquema mostrado na Figura 1:

  • A tabela Customers registra os nomes de cada cliente.
  • As tabelas Orders registram todos os pedidos feitos.
  • A tabela Products armazena as informações de cada produto.

Há duas relações de chave externa entre essas tabelas:

  • Um relacionamento de chave externa é definido entre a tabela Orders e a tabela Customers para garantir que um pedido não seja criado a menos que haja um cliente correspondente.

  • Uma relação de chave externa entre a tabela Orders e a tabela Products garante que não seja possível criar um pedido para um produto que não existe.

Usando o esquema anterior como exemplo, este tópico discute as instruções da linguagem de definição de dados (DDL) CONSTRAINT que podem ser usadas para gerenciar relações entre tabelas em um banco de dados.

Por padrão, todas as chaves estrangeiras no Spanner são chaves estrangeiras aplicadas, que aplicam a integridade referencial. No Spanner, você também pode usar chaves estrangeiras informativas, que não validam nem aplicam a integridade referencial. Para mais informações, consulte Comparação de chaves externas e Escolher o tipo de chave externa a ser usado. Quando não especificado, as chaves estrangeiras nos exemplos desta página são chaves estrangeiras aplicadas.

Adicionar uma chave estrangeira a uma nova tabela

Suponha que você tenha criado uma tabela Customers no banco de dados básico de pedidos de produtos. Agora, você precisa de uma tabela Orders para armazenar informações sobre os pedidos que os clientes fazem. Para garantir que todos os pedidos sejam válidos, não permita que o sistema insira linhas na tabela Orders, a menos que haja também uma entrada correspondente na tabela Customers. Portanto, você precisa de uma chave estrangeira aplicada para estabelecer uma relação entre as duas tabelas. Uma opção é adicionar uma coluna CustomerID à nova tabela e usá-la como chave externa para criar uma relação com a coluna CustomerID na tabela Customers.

Ao criar uma tabela com uma chave estrangeira, use REFERENCE para estabelecer uma relação com outra tabela. A tabela que contém a instrução REFERENCE é chamada de tabela de referência. A tabela nomeada na instrução REFERENCE é a tabela referenciada. A coluna nomeada na instrução REFERENCE é chamada de coluna de referência.

O exemplo a seguir mostra como usar a instrução DDL CREATE TABLE para criar a tabela Orders com uma restrição de chave externa que faz referência a CustomerID na tabela Customers.

GoogleSQL

CREATE TABLE Orders (
  OrderID INT64 NOT NULL,
  CustomerID INT64 NOT NULL,
  Quantity INT64 NOT NULL,
  ProductID INT64 NOT NULL,
  CONSTRAINT FK_CustomerOrder FOREIGN KEY (CustomerID) REFERENCES Customers (CustomerID)
) PRIMARY KEY (OrderID);

PostgreSQL

CREATE TABLE Orders (
  OrderID BIGINT NOT NULL,
  CustomerID BIGINT NOT NULL,
  Quantity BIGINT NOT NULL,
  ProductID BIGINT NOT NULL,
  CONSTRAINT FK_CustomerOrder FOREIGN KEY (CustomerID) REFERENCES Customers (CustomerID),
  PRIMARY KEY (OrderID)
);

A instrução anterior contém uma cláusula CONSTRAINT com as seguintes características:

  • O uso da sintaxe CONSTRAINT para nomear uma restrição, facilitando a remoção da tabela usando o nome escolhido.

  • A restrição tem o nome FK_CustomerOrder. Os nomes das restrições têm escopo no esquema e precisam ser exclusivos dentro dele.

  • A tabela Orders, em que você define a restrição, é a tabela de referência. A tabela Customers é a tabela referenciada.

  • A coluna de referência na tabela de referência é CustomerID. Ela faz referência ao campo CustomerID na tabela Customers. Se alguém tentar inserir uma linha em Orders com um CustomerID que não existe em Customers, a inserção vai falhar.

O exemplo a seguir mostra uma instrução de criação de tabela alternativa. Aqui, a restrição de chave externa é definida sem um nome. Quando você usa essa sintaxe, o Spanner gera um nome para você. Para descobrir os nomes de todas as chaves externas, consulte Exibir propriedades de uma relação de chave externa.

GoogleSQL

CREATE TABLE Orders (
  OrderID INT64 NOT NULL,
  CustomerID INT64 NOT NULL,
  ProductID INT64 NOT NULL,
  FOREIGN KEY (CustomerID) REFERENCES Customers (CustomerID)
) PRIMARY KEY (OrderID);

PostgreSQL

CREATE TABLE Orders (
  OrderID BIGINT NOT NULL,
  CustomerID BIGINT NOT NULL,
  Quantity BIGINT NOT NULL,
  ProductID BIGINT NOT NULL,
  FOREIGN KEY (CustomerID) REFERENCES Customers (CustomerID),
  PRIMARY KEY (OrderID)
);

Adicionar uma chave estrangeira a uma tabela atual

Você também quer garantir que os clientes só possam pedir produtos que existem. Se a tabela tiver restrições, remova todas elas. No Spanner, todas as restrições aplicadas em uma tabela precisam ser implementadas ao mesmo tempo em uma única instrução DDL em lote.

Se a tabela não tiver restrições, use a instrução DDL ALTER TABLE para adicionar uma restrição de chave externa aplicada à tabela Orders, conforme mostrado no exemplo a seguir:

ALTER TABLE Orders
  ADD CONSTRAINT DB_ProductOrder FOREIGN KEY (ProductID) REFERENCES Products (ProductID);

A coluna de referência em Orders é ProductID e faz referência à coluna ProductID em Products. Se você estiver bem com o Spanner nomeando essas restrições para você, use a seguinte sintaxe:

ALTER TABLE Orders
  ADD FOREIGN KEY (ProductID) REFERENCES Products (ProductID);

Adicionar uma chave estrangeira com uma ação de exclusão a uma nova tabela

Lembre-se do exemplo anterior em que você tem uma tabela Customers em um banco de dados de pedidos de produtos que precisa de uma tabela Orders. Você quer adicionar uma restrição de chave externa que faça referência à tabela Customers. No entanto, você quer garantir que, quando excluir um registro de cliente no futuro, o Spanner também exclua todos os pedidos desse cliente. Nesse caso, use a ação ON DELETE CASCADE com a restrição de chave externa.

A instrução DDL CREATE TABLE a seguir para a tabela Orders inclui a restrição de chave externa que faz referência à tabela Customers com uma ação ON DELETE CASCADE.

GoogleSQL

CREATE TABLE Orders (
  OrderID INT64 NOT NULL,
  CustomerID INT64 NOT NULL,
  Quantity INT64 NOT NULL,
  ProductID INT64 NOT NULL,
  CONSTRAINT FK_CustomerOrder FOREIGN KEY (CustomerID)
    REFERENCES Customers (CustomerID) ON DELETE CASCADE
) PRIMARY KEY (OrderID);

PostgreSQL

CREATE TABLE Orders (
  OrderID BIGINT NOT NULL,
  CustomerID BIGINT NOT NULL,
  Quantity BIGINT NOT NULL,
  ProductID BIGINT NOT NULL,
  FOREIGN KEY (CustomerID)
    REFERENCES Customers (CustomerID) ON DELETE CASCADE,
  PRIMARY KEY (OrderID)
);

A instrução anterior contém uma restrição de chave externa com uma cláusula ON DELETE CASCADE. A coluna CustomerID é uma chave externa que faz referência ao campo CustomerID na tabela Customers. Isso significa que cada valor CustomerID na tabela Orders também precisa existir na tabela Customers. Se alguém tentar excluir uma linha da tabela Customers, todas as linhas da tabela Orders que fazem referência ao valor CustomerID excluído também serão excluídas na mesma transação.

Adicionar uma chave estrangeira com uma ação de exclusão a uma tabela

Também queremos garantir que só possam ser criados pedidos de produtos existentes. Você pode usar ALTER TABLE para adicionar outra restrição de chave externa com ação ON DELETE CASCADE à tabela de pedidos da seguinte maneira:

ALTER TABLE Orders
  ADD CONSTRAINT DB_ProductOrder FOREIGN KEY (ProductID)
    REFERENCES Products (ProductID) ON DELETE CASCADE;

Excluir uma linha da tabela Products remove todas as linhas da tabela Orders que fazem referência ao valor ProductID excluído.

Usar chaves estrangeiras informativas (somente GoogleSQL)

As chaves externas informativas permitem que o otimizador de consultas use a relação de chave externa sem a sobrecarga incorrida pelas verificações de integridade referencial realizadas pelas chaves externas obrigatórias. As chaves estrangeiras informativas são úteis quando a aplicação de uma integridade referencial estrita é impraticável ou gera uma sobrecarga significativa de performance.

Continuando com o exemplo anterior, imagine que você quer modelar as relações entre as tabelas Customers, Orders e Products. No entanto, aplicar uma integridade referencial estrita nos dados das tabelas pode gerar gargalos de desempenho, especialmente durante os períodos de pico de compras com altos volumes de pedidos. Além disso, os clientes podem fazer pedidos de produtos que foram descontinuados e removidos da tabela Products.

É possível criar a tabela Orders usando chaves estrangeiras informativas:

CREATE TABLE Orders (
    OrderID INT64 NOT NULL,
    CustomerID INT64 NOT NULL,
    Quantity INT64 NOT NULL,
    ProductID INT64 NOT NULL,
    CONSTRAINT FK_CustomerOrder FOREIGN KEY (CustomerID) REFERENCES Customers (CustomerID) NOT ENFORCED,
    CONSTRAINT FK_ProductOrder FOREIGN KEY (ProductID) REFERENCES Products (ProductID) NOT ENFORCED
) PRIMARY KEY (OrderID);

Ao criar uma chave estrangeira informativa com NOT ENFORCED, você permite a possibilidade de um pedido fazer referência a um cliente ou produto inexistente. Usar uma chave externa informativa em vez de uma restrição de chave externa é uma boa opção se uma conta de cliente puder ser excluída ou um produto descontinuado. Com uma chave externa informativa, o Spanner não realiza a validação de integridade referencial. Isso reduz a sobrecarga de gravação, melhorando potencialmente o desempenho durante os horários de pico de processamento de pedidos.

Você pode permitir que o otimizador de consultas use as relações para gerar planos de consulta eficientes. Isso pode melhorar a performance das consultas que combinam as tabelas em colunas de chave estrangeira. Para mais informações, consulte chave externa informativa para otimização de consultas.

Consultar dados em relações de chave externa

SELECT * FROM Orders
  INNER JOIN Customers ON Orders.CustomerID = Customers.CustomerID
  INNER JOIN Products ON Orders.ProductsID = Products.ProductID;

Integridade referencial com chaves estrangeiras aplicadas

O principal motivo para adicionar relações de chave externa forçadas é para que o Spanner possa manter a integridade referencial dos dados. Se você modificar os dados de uma maneira que interrompa uma restrição de chave externa, a atualização vai falhar com um erro.

Considere os dados na Figura 2. Alguns clientes pediram produtos, conforme mostrado na tabela Orders. Devido à restrição de chave externa aplicada em vigor, os dados inseridos na tabela Orders têm integridade referencial.

Dados de amostra das tabelas "Clientes", "Produtos" e "Pedidos".

Figura 2. Dados de amostra do nosso banco de dados de pedidos.

Os exemplos a seguir mostram o que acontece quando você tenta modificar os dados de uma forma que viola a integridade referencial.

  • Adicionar uma linha à tabela Orders com um valor CustomerID que não existe em Customers

    O que acontece se tentarmos a seguinte modificação, considerando os dados de amostra do diagrama anterior?

    INSERT INTO Orders (OrderID, ProductID, Quantity, CustomerID)
      VALUES (19, 337876, 4, 447);
    

    Nesse caso, o sistema tentaria inserir uma linha em Orders com um CustomerID (447) que não existe na tabela Customers. Se o sistema fizesse isso, você teria um pedido inválido. No entanto, com a restrição de chave externa aplicada que você adicionou à tabela Orders, ela está protegida. O INSERT falha com a seguinte mensagem, supondo que a restrição se chame FK_CustomerOrder.

    Foreign key constraint `FK_CustomerOrder` is violated on table `Orders`.
    Cannot find referenced values in Customers(CustomerID).
    

    Ao contrário das chaves externas aplicadas, as chaves externas informativas não aplicam a integridade referencial. Se FK_CustomerOrder for uma chave externa informativa, a instrução de inserção será bem-sucedida porque o Spanner não valida se o CustomerID correspondente existe na tabela Customers. Por isso, os dados podem não estar em conformidade com a integridade referencial definida por FK_CustomerOrder.

  • Tentar excluir uma linha da tabela Customers quando o cliente é referenciado em uma restrição de chave externa aplicada.

    Imagine uma situação em que um cliente cancela a inscrição da sua loja on-line. Você quer remover o cliente do seu back-end, então tenta a seguinte operação.

    DELETE FROM Customers WHERE CustomerID = 721;
    

    Neste exemplo, o Spanner detecta, usando a restrição de chave externa, que ainda há registros na tabela Orders que fazem referência à linha do cliente que você está tentando excluir. O erro a seguir é exibido nesse caso.

    Foreign key constraint violation when deleting or updating referenced row(s): referencing row(s) found in table `Orders`.

    Para corrigir esse problema, exclua todas as entradas de referência em Orders primeiro. Também é possível definir a chave estrangeira com a ação ON DELETE CASCADE para permitir que o Spanner processe a exclusão de entradas de referência.

    Da mesma forma, se FK_CustomerOrder for uma chave estrangeira informativa, a ação de exclusão será bem-sucedida porque o Spanner não garante a integridade referencial de chaves estrangeiras informativas.

Ver propriedades de um relacionamento de chave estrangeira

A INFORMATION_SCHEMA do Spanner contém informações sobre chaves externas e os respectivos índices de backup. Confira alguns exemplos de perguntas que você pode responder consultando o INFORMATION SCHEMA.

Para mais informações sobre índices de backup, consulte Índices de backup de chaves externas.

Quais restrições são definidas no meu banco de dados?

SELECT tc.CONSTRAINT_NAME, tc.TABLE_NAME, tc.CONSTRAINT_TYPE
FROM INFORMATION_SCHEMA.TABLE_CONSTRAINTS as tc
WHERE tc.CONSTRAINT_TYPE = 'FOREIGN KEY';

Quais chaves externas são definidas no meu banco de dados?

SELECT rc.CONSTRAINT_NAME, rc.UNIQUE_CONSTRAINT_NAME, rc.SPANNER_STATE
FROM INFORMATION_SCHEMA.REFERENTIAL_CONSTRAINTS as rc;

Quais índices são secundários para chaves estrangeiras, também conhecidos como índices de backup?

Os índices de backup de chave externa são gerenciados pelo Spanner. Portanto , consultar SPANNER_IS_MANAGED na visualização INDEXES retorna todos os índices de backup.

SELECT i.TABLE_NAME, i.INDEX_NAME, i.INDEX_TYPE, i.INDEX_STATE,
  i.IS_UNIQUE, i.IS_NULL_FILTERED, i.SPANNER_IS_MANAGED
FROM INFORMATION_SCHEMA.INDEXES as i
WHERE SPANNER_IS_MANAGED = 'YES';

Qual é a ação referencial definida com a restrição de chave externa?

SELECT rc.CONSTRAINT_NAME, rc.UNIQUE_CONSTRAINT_NAME, rc.DELETE_RULE,
  rc.UPDATE_RULE
FROM INFORMATION_SCHEMA.REFERENTIAL_CONSTRAINTS as rc;

Uma chave externa é aplicada ou não aplicada?

SELECT tc.CONSTRAINT_NAME, tc.TABLE_NAME, tc.CONSTRAINT_TYPE, tc.ENFORCED
FROM INFORMATION_SCHEMA.TABLE_CONSTRAINTS as tc
WHERE tc.CONSTRAINT_TYPE = 'FOREIGN KEY';

Para mais informações, consulte Esquema de informações.

Remover um relacionamento de chave estrangeira

A DDL a seguir remove uma restrição de chave externa da tabela Orders.

ALTER TABLE Orders
  DROP CONSTRAINT FK_CustomerOrder;

Os índices de backup de chave externa são descartados automaticamente quando a própria restrição é descartada.

Suporte para relacionamentos mais complexos de chaves estrangeiras

Os tópicos a seguir mostram como usar chaves estrangeiras para aplicar relações mais complexas entre tabelas.

Várias colunas

As chaves externas podem fazer referência a várias colunas. A lista de colunas forma uma chave que corresponde à chave primária de uma tabela ou a um índice de backup. A tabela de referência contém chaves externas da chave de tabela referenciada.

No exemplo a seguir, as definições de chave externa aplicadas indicam que:

  • Cada valor SongName na tabela TopHits precisa ter um valor correspondente na tabela Songs.

  • Cada par de valores SingerFirstName e SingerLastName precisa ter um par correspondente de valores FirstName e LastName na tabela Singers.

GoogleSQL

CREATE TABLE TopHits (
  Rank INT64 NOT NULL,
  SongName STRING(MAX),
  SingerFirstName STRING(MAX),
  SingerLastName STRING(MAX),

  -- Song names must either be NULL or have matching values in Songs.
  FOREIGN KEY (SongName) REFERENCES Songs (SongName),

  -- Singer names must either be NULL or have matching values in Singers.
  FOREIGN KEY (SingerFirstName, SingerLastName)
  REFERENCES Singers (FirstName, LastName)

) PRIMARY KEY (Rank);

PostgreSQL

CREATE TABLE TopHits (
  Rank BIGINT NOT NULL,
  SongName VARCHAR,
  SingerFirstName VARCHAR,
  SingerLastName VARCHAR,

  -- Song names must either be NULL or have matching values in Songs.
  FOREIGN KEY (SongName) REFERENCES Songs (SongName),

  -- Singer names must either be NULL or have matching values in Singers.
  FOREIGN KEY (SingerFirstName, SingerLastName)
  REFERENCES Singers (FirstName, LastName),

  PRIMARY KEY (Rank)
);

Referências circulares

Ocasionalmente, as tabelas têm dependências circulares, talvez por motivos legados ou devido à desnormalização. As chaves externas do Spanner permitem referências circulares. Como uma tabela referenciada precisa existir antes que uma chave externa possa referenciá-la, uma das chaves externas precisa ser adicionada com uma instrução ALTER TABLE. Veja um exemplo:

  1. Crie TableA, sem uma chave externa.
  2. Crie TableB com uma restrição de chave estrangeira em TableA.
  3. Use ALTER TABLE em TableA para criar uma referência de chave externa para TableB.

Tabelas de autorreferência

Um tipo especial de referência circular é uma tabela que define uma chave externa que faz referência à mesma tabela. Por exemplo, o snippet a seguir mostra uma chave externa para determinar que o ManagerId de um funcionário também seja um funcionário.

GoogleSQL

CREATE TABLE Employees (
  EmployeeId INT64 NOT NULL,
  EmployeeName STRING(MAX) NOT NULL,
  ManagerId INT64,
  FOREIGN KEY (ManagerId) REFERENCES Employees (EmployeeId)
) PRIMARY KEY (EmployeeId);

PostgreSQL

CREATE TABLE Employees (
  EmployeeId BIGINT NOT NULL,
  EmployeeName VARCHAR NOT NULL,
  ManagerId BIGINT,
  FOREIGN KEY (ManagerId) REFERENCES Employees (EmployeeId),
  PRIMARY KEY (EmployeeId)
);

A seguir