AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / dba / Perguntas / 172373
Accepted
JBentley
JBentley
Asked: 2017-05-01 08:22:46 +0800 CST2017-05-01 08:22:46 +0800 CST 2017-05-01 08:22:46 +0800 CST

Como modelar um relacionamento com tabelas de variáveis ​​(polimorfismo)?

  • 772

Qual é a melhor maneira de modelar o seguinte relacionamento de tipo?

Digamos que eu tenha tabelas representando dois tipos de entidades: companye person. Eu tenho uma customertabela com um entity_idcampo, que deve se referir a a companyou a person.

Se eu simplesmente armazenar a company_idou a , não saberei em qual tabela procurar a entidade. A primeira solução que me ocorreu é ter um person_idcampo no qual levaria os valores "empresa" ou "pessoa" para diga-me a qual tabela minha chave estrangeira se refere. Não tenho certeza se esta é a melhor maneira embora.entity_identity_typecustomerentity_id

Isso parece ser um problema bastante comum no design de banco de dados, então espero que haja um padrão de design que eu possa adotar.

Eu usei a palavra polimorfismo no título no sentido mais amplo, então não estou me referindo à herança. No meu exemplo, uma empresa não é uma pessoa (ou vice-versa), mas ainda são intercambiáveis. Outras perguntas que encontrei nos sites do Stack Exchange assumiram que a herança também é um requisito, portanto, não tenho certeza de que haja uma duplicata adequada.

database-design design-pattern
  • 2 2 respostas
  • 3168 Views

2 respostas

  • Voted
  1. Best Answer
    stefan
    2017-05-01T10:55:04+08:002017-05-01T10:55:04+08:00

    Parece haver 2 maneiras principais de modelar tal situação:

    (1) Use supertipos/subtipos. No entanto, como não estamos discutindo design "orientado a objetos", a segunda opção pode ser um pouco mais apropriada.

    (2) Use o chamado ARC, que expressa uma relação XOR.

    Suponha que temos uma entidade TRANSACTION. Cada cliente pode estar envolvido em uma ou mais transações E um cliente é uma EMPRESA ou uma PESSOA (XOR). Nosso modelo ERD/relacional pode ser parecido com os abaixo. Observe o "arco", que é desenhado em ambos os relacionamentos. insira a descrição da imagem aqui

    insira a descrição da imagem aqui

    Quanto à implementação: você pode codificar uma restrição CHECK (para evitar triggers), que irá impor o XOR (ou exclusivo). O código DDL completo se parece com isso (observe o CHECK):

    CREATE TABLE company ( 
      c_id NUMBER PRIMARY KEY
    );
    
    CREATE TABLE person ( 
      p_id NUMBER PRIMARY KEY
    );
    
    CREATE TABLE transaction ( 
      t_id NUMBER PRIMARY KEY
    , c_id NUMBER 
    , p_id NUMBER
    );
    
    ALTER TABLE transaction 
    ADD CONSTRAINT fkarc CHECK (
         ( (c_id IS NOT NULL) AND (p_id IS NULL) ) 
      OR ( (p_id IS NOT NULL) AND (c_id IS NULL) ) 
    );
    
    ALTER TABLE transaction
    ADD CONSTRAINT transaction_company_fk 
    FOREIGN KEY ( c_id ) REFERENCES COMPANY ( c_id ) ;
    
    ALTER TABLE transaction
    ADD CONSTRAINT transaction_person_fk
    FOREIGN KEY ( p_id ) REFERENCES PERSON ( p_id ) ;
    

    (Testado no Oracle 12c)

    • 1
  2. Walter Mitty
    2017-05-02T06:53:36+08:002017-05-02T06:53:36+08:00

    Eu sei que você disse que não está se referindo à herança, mas a melhor literatura sobre o seu caso se refere à herança ao descrever o problema e propor uma solução.

    Há um punhado de sites que fornecem soluções detalhadas sobre como projetar tabelas relacionais que expressam o tipo/subtipo ou o relacionamento classe/subclasse. O mais importante é usar as palavras-chave certas em sua pesquisa.

    Na modelagem Entidade-Relacionamento entre clientes, empresas e pessoas é chamada de "generalização/especialização". Uma pesquisa sobre isso lhe dará muito material bom. As empresas são clientes especializados, assim como as pessoas. outra palavra-chave útil é "relacionamentos IS-A".

    Na modelagem relacional, existem três técnicas amplamente compartilhadas para cobrir esse tipo de problema. Aqui eles são "herança de tabela única", "herança de tabela de classe" e "chave primária compartilhada". Eu sei que a palavra herança aparece lá, mas acredite em mim, uma pesquisa sobre esses itens fornecerá alguns artigos que abordam seu caso específico, embora possam estar falando sobre automóveis, caminhões e veículos ou cães, gatos e animais de estimação. de empresas, pessoas e clientes.

    Alguns dos acessos que você receberá o levarão de volta aqui, para stackexchange.dba. Outros o levarão ao Stackoverflow, onde há três tags para coletar informações sobre as três técnicas.

    A única coisa não coberta em detalhes é como você adiciona restrições para impor a regra de exclusão mútua, o que você chamou de XOR. Se você usar a chave primária compartilhada, a restrição para exclusão mútua será muito parecida com a que você já apresentou, só que mais simples.

    Para resumir o que tudo isso significa no seu caso:

    Você terá quatro tabelas: transação, cliente, pessoa e empresa.

    A transação conterá um campo customer_id, que faz referência ao campo id em customer. O cliente terá um campo de identificação, provavelmente preenchido por um recurso de numeração automática, como é feito normalmente. A pessoa não tem um campo id. Em vez disso, ele tem um campo company_id que atua como uma chave primária e uma chave estrangeira que faz referência a company.id. A empresa é tratada da mesma forma que a pessoa, com um campo company_id que é um PK e um FK.

    A tabela de clientes pode conter campos, se houver, que pertencem a empresas e pessoas.

    Por fim, você precisará de uma restrição na tabela person que diga que customer_id não está na tabela company e uma na tabela company que diga que customer_id não está na tabela person.

    • 1

relate perguntas

  • Os índices filtrados podem ajudar a melhorar as consultas baseadas em uma hora inserida ou isso deve ser evitado?

  • Qual é a diferença entre os tipos de dados MySQL VARCHAR e TEXT?

  • É melhor armazenar os valores calculados ou recalculá-los a pedido? [duplicado]

  • Armazenar vs calcular valores agregados

  • Quais são algumas maneiras de implementar um relacionamento muitos-para-muitos em um data warehouse?

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    conectar ao servidor PostgreSQL: FATAL: nenhuma entrada pg_hba.conf para o host

    • 12 respostas
  • Marko Smith

    Como fazer a saída do sqlplus aparecer em uma linha?

    • 3 respostas
  • Marko Smith

    Selecione qual tem data máxima ou data mais recente

    • 3 respostas
  • Marko Smith

    Como faço para listar todos os esquemas no PostgreSQL?

    • 4 respostas
  • Marko Smith

    Listar todas as colunas de uma tabela especificada

    • 5 respostas
  • Marko Smith

    Como usar o sqlplus para se conectar a um banco de dados Oracle localizado em outro host sem modificar meu próprio tnsnames.ora

    • 4 respostas
  • Marko Smith

    Como você mysqldump tabela (s) específica (s)?

    • 4 respostas
  • Marko Smith

    Listar os privilégios do banco de dados usando o psql

    • 10 respostas
  • Marko Smith

    Como inserir valores em uma tabela de uma consulta de seleção no PostgreSQL?

    • 4 respostas
  • Marko Smith

    Como faço para listar todos os bancos de dados e tabelas usando o psql?

    • 7 respostas
  • Martin Hope
    Jin conectar ao servidor PostgreSQL: FATAL: nenhuma entrada pg_hba.conf para o host 2014-12-02 02:54:58 +0800 CST
  • Martin Hope
    Stéphane Como faço para listar todos os esquemas no PostgreSQL? 2013-04-16 11:19:16 +0800 CST
  • Martin Hope
    Mike Walsh Por que o log de transações continua crescendo ou fica sem espaço? 2012-12-05 18:11:22 +0800 CST
  • Martin Hope
    Stephane Rolland Listar todas as colunas de uma tabela especificada 2012-08-14 04:44:44 +0800 CST
  • Martin Hope
    haxney O MySQL pode realizar consultas razoavelmente em bilhões de linhas? 2012-07-03 11:36:13 +0800 CST
  • Martin Hope
    qazwsx Como posso monitorar o andamento de uma importação de um arquivo .sql grande? 2012-05-03 08:54:41 +0800 CST
  • Martin Hope
    markdorison Como você mysqldump tabela (s) específica (s)? 2011-12-17 12:39:37 +0800 CST
  • Martin Hope
    Jonas Como posso cronometrar consultas SQL usando psql? 2011-06-04 02:22:54 +0800 CST
  • Martin Hope
    Jonas Como inserir valores em uma tabela de uma consulta de seleção no PostgreSQL? 2011-05-28 00:33:05 +0800 CST
  • Martin Hope
    Jonas Como faço para listar todos os bancos de dados e tabelas usando o psql? 2011-02-18 00:45:49 +0800 CST

Hot tag

sql-server mysql postgresql sql-server-2014 sql-server-2016 oracle sql-server-2008 database-design query-performance sql-server-2017

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve