- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我很少遇到这样的情况,多个实体类型的单个表似乎比每个实体类型一个表更好。这是一个对我有意义的例子,但在学术上似乎是错误的。
问题:我可以这样做并且仍然拥有“声音”架构吗?
示例如下
假设有两种实体类型,一个公司和一个人。一家公司通常由一个人所有,但有时另一家公司拥有一家公司。
坚持这个想法,并补充说,假设每个公司都有一个注册代理人,负责公司的合法创建。进一步说明,注册代理人可以是个人或公司。
如果您认为公司 [子] 的所有者 [母公司] 可以是个人或公司,您可能会开始看到保持第三范式和避免冗余的挑战。
与我的示例相反,如果只有人可以拥有 Corporations,那么 Ownership 链接表非常常规,具有以下列:OwnershipID(有点不必要)、CorporationID、PersonID。
相反,您需要类似的信息:OwnershipID、CorporationID、OwnerID、OwnerEntityType(公司或个人)。不要误会我的意思,你可以让这个工作,但至少可以说它不会很有趣。
继续我给出的示例,您需要为每个公司分配一个代理。通常,代理人是所有者之一(一个人)。在这种情况下,您确实希望链接回该人的一个记录。您不想将该人记录为所有者然后再次作为代理(在代理表中)。那将是多余的。
与该“问题”类似,注册代理人也可以是公司,例如律师事务所、注册会计师或商业文件公司,举一些典型的例子。就像代理人一样,代理人公司真的不应该获得自己作为代理人实体的记录。相反,它需要链接回 Corporation 表中其公司存在的 。 [除了我最终说没有 CorporationEntity 表]
就像将每个公司与其任何类型、个人或公司的所有者相匹配的链接表一样,您可以拥有一个代理链接表:AgentRepresentationID、CorporationID、AgentID、AgentType……但同样,它会很丑( IMO) 当您必须将相关代理召集在一起时——一些来自 Person 表,一些来自 Corporation 表。
这就是为什么我说,在这种情况下,您可以看到中性实体类型如何具有优势。它会是这样的:
表:实体全部
关键列:EntityID、EntityType(或 EntityTypeID,如果您坚持,请链接出来以获取描述)、EntityName(有关于名称和不同类型的问题......本文的主题之外)
链接表:公司所有权
关键列:OwnershipID(同样,我认为这是不必要的),ChildEntityID(被拥有的实体;为了清楚起见,命名为“Child”,我不会这样命名)ParentEntityID(父实体)
链接表:代理代表
关键列:AgentRepresentationID(...我不会说)、CorporationEntityID(被表示的公司实体)、AgentEntityID(来自实体表,相当于这里的代理记录)
虽然您可能对我的架构没意见,但您应该对链接表中的列命名有些困扰。这让我很烦。通常,这些表中的第二个和第三个列名与您在每个实体的各自表中加入的列的名称完全匹配(哈哈,但每个实体都没有各自的表,因此您不能让链接表列名匹配源列名称,因为它们是同一列)。从技术上讲,这并不重要,但它会破坏您的命名约定,这很重要,但还不足以不这样做。
如果我还没有把它开回家,这里是你将它拉到一起的方法。您自己加入 EntityAll 表以获得您需要的内容。
列出所有军团及其所有者(在 T-SQL 中):
SELECT Corp.EntityName as CorpName, Owner.EntityName as OwnerName
FROM EntityAll as Corp
JOIN CorporationOwnership as Link on (Corp.EntityID = Link.ChildEntityID)
JOIN EntityAll as Owner on (Link.ParentEntityID = Owner.EntityID)
最佳答案
我认为是 Hugh Darwen 创造了术语“分布式键”和“分布式外键”,其中一个被引用的键值恰好存在于多个引用相关变量(表)之一中;这将需要一个相关的“多重赋值”概念,以便原子地插入到引用和引用的 relvars。
虽然这在理论上可以在 SQL-92 中使用延迟模式级实现 ASSERTION
s(或者可能是 CHECK
支持子查询的约束),这是一个相当笨重的过程,是程序性的(而不是基于集合的),并且没有支持此功能的 SQL 产品(或者永远不会,我怀疑) .
对于可用的 SQL 产品,我们能做的最好的事情是使用复合键 (entity_ID, entity_type)
与 CHECK
entity_type
上的约束在引用表中以确保引用键值不超过一个(请注意,这与“恰好一个引用键值”不同),例如
CREATE TABLE LegalPersons
(
person_ID INTEGER IDENTITY NOT NULL UNIQUE,
person_type VARCHAR(14) NOT NULL
CHECK (person_type IN ('Company', 'Natural Person')),
UNIQUE (person_type, person_ID)
);
CREATE TABLE Companies
(
person_ID INTEGER NOT NULL UNIQUE,
person_type VARCHAR(14) NOT NULL
CHECK (person_type = 'Company'),
FOREIGN KEY (person_type, person_ID)
REFERENCES LegalPersons (person_type, person_ID),
companies_house_registered_number VARCHAR(8) NOT NULL UNIQUE
-- other company columns and constraints here
);
CREATE TABLE NaturalPersons
(
person_ID INTEGER NOT NULL UNIQUE,
person_type VARCHAR(14) NOT NULL
CHECK (person_type = 'Natural Person'),
FOREIGN KEY (person_type, person_ID)
REFERENCES LegalPersons (person_type, person_ID)
-- natural person columns and constraints here
);
CREATE TABLE CompanyAgents
(
company_person_ID INTEGER NOT NULL UNIQUE,
company_person_type VARCHAR(14) NOT NULL
CHECK (company_person_type = 'Company'),
FOREIGN KEY (company_person_type, company_person_ID)
REFERENCES LegalPersons (person_type, person_ID),
agent_person_ID INTEGER NOT NULL,
agent_person_type VARCHAR(14) NOT NULL,
FOREIGN KEY (agent_person_type, agent_person_ID)
REFERENCES LegalPersons (person_type, person_ID),
CHECK (company_person_ID <> agent_person_ID)
);
agent_person_ID
使用单列键例如
agent_person_ID INTEGER NOT NULL
REFERENCES LegalPersons (person_ID)
JOIN
在 SQL DML 中 :)
关于SQL 架构 : Is this a justified case to have only one table storing multiple entity types?(使用自联接),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3827549/
我经常在 ATS 中看到使用 case、case+ 或 case- 形成的 case 表达式。有什么区别? 最佳答案 如果表达式不详尽,使用 case 会发出警告,case+ 会产生错误,而 case
我有一个导入时全部大写的表,我想将其转换为正确的大小写。你们用什么脚本来完成这个? 最佳答案 这个函数: “正确大小写”由空格分隔的所有“大写”单词 保留“小写单词” 即使对于非英语字母也能正常工作
#include int main() { switch(2) { case 1: if(1)
我已经四处寻找了一段时间,如果我使用的术语不当,请原谅我... 代码的目标是在输入为 0 时更新 Aout1 和 Aout0,输出对应于 7 段显示,但出现以下错误: “错误 (10170):Four
我正在尝试按照 PostgreSQL 手册中的说明进行操作。 PostgreSQL: Documentation: 9.1: Control Structures 我的 PostgreSQL 服务器是
我有一个状态机,其中有几个非常相似的状态。我可以为每个状态编写它,如下例所示: module CHECK_FSM ( GO, DONE, CLK, RESETN ); input GO;
如何使用或创建案例? 就像是: string str; case (str) "abc" || "dfg": begin //some code end "yfg":
这个问题已经有答案了: Are double and single quotes interchangeable in JavaScript? (23 个回答) 已关闭 9 年前。 我正在学习Java
汽车 Make | Model | Year | Color Honda | Accord | 12 | Red Lexus | IS | 14 |
如何使用当前 case 语句的值跳转到 switch-case 条件下的另一个 case 语句? 是否可以使用 switch case 来实现这种事情,或者是否有其他实现方式? 有可能实现吗?如果没有
我理解下面的代码。 var day = 2; switch (day) { case 1: document.write("Monday"); break;
这是有效的。 object FilesToDFDS { case class Student(id: Int, name: String, dept:String) def main(
我对 VHDL 还是个新手。我需要在 CASE 语句中为多个信号赋值,如下所示: CASE input24 IS WHEN "00" THEN output0
我有这个 case 语句,它给出了一个错误“变量 constant1 未使用”。它似乎忽略了变量并返回了第一行,因此变量显然没有范围。如果我用数字 1 替换常量,那么它就可以工作。在 Elixir 中
在 MySQL 中,是否可以在 SELECT 子句中有两个 CASE 语句,其中第二个 CASE 语句依赖于第一个 CASE 语句? 例如,考虑以下查询: SELECT CASE WHEN `user
我正在尝试一个挑战,我需要获得一个随机数,并在没有重复的情况下打印数字内的数字总和:例如,123 将打印 6 ( 1 + 2 + 3 ),而 32111 将做同样的事情(因为我们没有在我们的总和中添加
当有人试图更新当前未存储在我的散列中的值时,我想立即返回 when 'add' 而无需重新启动整个 case声明,因为我已经知道他们想要添加并且不想再次提示他们。 有没有一种方法可以在不重新启动整个案
老 C 程序员可以在 Swift 方面得到一些帮助。 我不太了解 if-case 语法。例如: if case 20...30 = age { print ("in range.") } cas
老 C 程序员可以在 Swift 方面得到一些帮助。 我不太了解 if-case 语法。例如: if case 20...30 = age { print ("in range.") } cas
我有一个 ArrayList,其中包含以下字符串:[name, age, gender, salary] . 有没有办法可以将 ArrayList 中的值用作 case 表达式? 显而易见的答案是否定
我是一名优秀的程序员,十分优秀!