SQL2005:将一个表链接到多个表并保留引用完整性?
这是我的数据库的简化:
Table: Property Fields: ID, Address Table: Quote Fields: ID, PropertyID, BespokeQuoteFields... Table: Job Fields: ID, PropertyID, BespokeJobFields...
然后我们还有其他分别与 Quote 和 Job 表相关的表。
我现在需要添加一个消息表,用户可以在其中记录客户留下的有关工作和报价的电话消息。
我可以创建两个相同的表(QuoteMessage 和 JobMessage),但这违反了 DRY 原则并且看起来很混乱。
我可以创建一个 Message 表:
Table: Message Fields: ID, RelationID, RelationType, OtherFields...
但这会阻止我使用约束来强制引用完整性。 我还可以预见它会在稍后使用 Linq to SQL 的开发端产生问题。
这个问题有一个优雅的解决方案吗,还是我最终不得不将一些东西组合在一起?
伯恩斯
Here is a simplification of my database:
Table: Property Fields: ID, Address Table: Quote Fields: ID, PropertyID, BespokeQuoteFields... Table: Job Fields: ID, PropertyID, BespokeJobFields...
Then we have other tables that relate to the Quote and Job tables individually.
I now need to add a Message table where users can record telephone messages left by customers regarding Jobs and Quotes.
I could create two identical tables (QuoteMessage and JobMessage), but this violates the DRY principal and seems messy.
I could create one Message table:
Table: Message Fields: ID, RelationID, RelationType, OtherFields...
But this stops me from using constraints to enforce my referential integrity. I can also forsee it creating problems with the devlopment side using Linq to SQL later on.
Is there an elegant solution to this problem, or am I ultimately going to have to hack something together?
Burns
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(4)
为什么不在消息表中同时包含 QuoteId 和 JobId 字段? 或者消息是否必须涉及报价或工作,而不是两者?
Why not just have both QuoteId and JobId fields in the message table? Or does a message have to be regarding either a quote or a job and not both?
我能想到的唯一其他方法是拥有一个基本消息表,其中包含 Id 和 TypeId。 然后,您的子表(QuoteMessage 和 JobMessage)引用 MessageId 和 TypeId 上的基表 - 但也对它们进行检查约束,以仅强制执行适当的 MessageTypeId。
与仅使用 JobMesssage 和 QuoteMessage 表相比,这会给您带来什么? 它将消息提升为一等公民,以便您可以从单个表中读取所有消息。 作为交换,您从消息到相关报价或工作的查询路径还需要 1 个联接。 这在某种程度上取决于您的应用程序流程,这是否是一个好的权衡。
至于两张相同的桌子违反了 DRY - 我不会对此耿耿于怀。 在数据库设计中,更多的是规范化,而不是 DRY。 如果您正在建模的两个事物具有相同的属性(列),但实际上是不同的事物(表) - 那么拥有多个具有相似模式的表是合理的。 比把不同的东西混在一起要好得多。
About the only other way I can think of is to have a base Message table, with both an Id and a TypeId. Your subtables (QuoteMessage and JobMessage) then reference the base table on both MessageId and TypeId - but also have CHECK CONSTRAINTS on them to enforce only the appropiate MessageTypeId.
What does this buy you, as compared to just a JobMesssage and QuoteMessage table? It elevates a Message to a first class citizen, so that you can read all Messages from a single table. In exchange, your query path from a Message to it's relevant Quote or Job is 1 more join away. It kind of depends on your app flow whether that's a good tradeoff or not.
As for 2 identical tables violating DRY - I wouldn't get hung up on that. In DB design, it's less about DRY, and more about normalization. If the 2 things you're modeling have the same attributes (columns), but are actually different things (tables) - then it's reasonable to have multiple tables with similar schemas. Much better than the reverse of munging different things together.
@burns
Ian 的回答 (+1) 是正确的[参见注释]。 使用多对多表
QUOTEMESSAGE
将QUOTE
连接到MESSAGE
是最正确的模型,但会留下孤立的MESSAGE
记录。这是可以使用触发器的罕见情况之一。 但是,需要小心以确保单个
MESSAGE
记录不能同时与QUOTE
和JOB
关联。Ian 请注意,我认为
JobMessage
的表定义中存在拼写错误,其中的列应为JobId、MessageId
(?)。 我会编辑你的引言,但我可能需要几年的时间才能获得这样的声誉!@burns
Ian's answer (+1) is correct [see note]. Using a many to many table
QUOTEMESSAGE
to joinQUOTE
toMESSAGE
is the most correct model, but will leave orphanedMESSAGE
records.This is one of those rare cases where a trigger can be used. However, caution needs to be applied to ensure that the a single
MESSAGE
record cannot be associated with both aQUOTE
and aJOB
.Note to Ian, I think there is a typo in the table definition for
JobMessage
, where the columns should beJobId, MessageId
(?). I would edit your quote but it might take me a few years to gain that level of reputation!创建一个消息表,其中包含唯一的 MessageId 以及需要为消息存储的各种属性。
创建两个链接表 - QuoteMessage 和 JobMessage。 这些将只包含两个字段,即报价/作业和消息的外键。
通过这种方式,您仅在一处定义了消息的数据属性(使其易于扩展和跨所有消息查询),但您还具有将报价和工作链接到任意数量的消息的引用完整性。 事实上,报价和工作都可以链接到相同消息(我不确定这是否适合您的业务模型,但至少数据模型为您提供了选择)。
Create one Message table, containing a unique MessageId and the various properties you need to store for a message.
Create two link tables - QuoteMessage and JobMessage. These will just contain two fields each, foreign keys to the Quote/Job and the Message.
In this way you have defined the data properties of a Message in one place only (making it easy to extend, and to query across all messages), but you also have the referential integrity linking Quotes and Jobs to any number of messages. Indeed, both a Quote and Job could be linked to the same message (I'm not sure if that is appropriate to your business model, but at least the data model gives you the option).