Если вы хотели полностью нормализованный дизайн, тогда у вас должно быть что-то вроде
Create Table dbo.Contacts
(
Id int not null Primary Key Clustered
, FirstName nvarchar(25) null
...
)
Create Table dbo.ContactAddresses
(
Id int not null Primary Key Clustered
, ContactId int not null
, Street nvarchar(max) null
, City nvarchar(40) null
...
, Constraint FK_ContactAddresses_Contacts
Foreign Key ( ContactId )
References dbo.Contacts( Id )
)
Create Table dbo.ContactEmail
(
Id int not null Primary Key Clustered
, ContactId int not null
, Type nvarchar(10) not null
, Address nvarchar(255) not null
...
, Constraint CK_ContactEmail_Type Check Type In('Primary','Secondary',...)
, Constraint UK_ContactEmail_Address Unique ( ContactId, Type, Address )
, Constraint FK_ContactEmail_Contacts
Foreign Key ( ContactId )
References dbo.Contacts( Id )
)
Create Table dbo.ContactPhone
(
Id int not null Primary Key Clustered
, ContactId int not null
, Type nvarchar(10) not null
, Number varchar(20) not null
...
, Constraint UK_ContactPhone Unique ( ContactId, Type, Number )
, Constraint CK_ContactPhone_Type Check Type In('Office','Fax','Home',...'Mobile')
, Constraint FK_ContactPhone_Contacts
Foreign Key ( ContactId )
References dbo.Contacts( Id )
)
Если вы хотите более контролируемый список типов электронной почты или телефонов, вы можете добавить родительскую таблицу и внешний ключ для их хранения. Чего не хватает в этом дизайне, так это защиты от адресации. Требуется ли пользователю вводить полный адрес? Этот дизайн позволяет неограниченное количество электронных писем, телефонных номеров и адресов. Однако это может быть излишним для того, что вам нужно.
Если вы разрабатываете полнофункциональный менеджер контактов для замены чего-то вроде Outlook, тогда вышеприведенный дизайн будет правильным решением. Тем не менее, большинство систем, которые собирают некоторые контактные данные, не предназначены для полного управления контактами. Если это так, то вам следует указать разумные ограничения на объем данных, которые вы будете собирать. Например, вы действительно собираетесь разрешить пользователям вводить восемь телефонных номеров для одного контакта? Вероятно, более реалистично, что 99% контактов будут иметь только два или три телефонных номера. Включите в контакт столбец nvarchar (max) Note для людей, которые хотят захватить сотни телефонных номеров и адресов электронной почты. Если это так, то вы можете немного денормализовать свой дизайн:
Create Table dbo.Contacts
(
Id int not null Primary Key Clustered
, FirstName nvarchar(25) null
...
, PrimaryEmailAddress nvarchar(255) null
, SecondaryEmailAddress nvarchar(255) null
, OfficePhone nvarchar(25) null
, MobilePhone nvarchar(25) null
, FaxPhone nvarchar(25) null
, Note nvarchar(max) null
...
, Constraint UK_Contacts Unique ( FirstName.... )
)
Единственный вопрос, который у вас есть на данный момент, касается адресов. Прежде чем я это сделаю, если руководство скажет: «Эй, мы хотим сохранить еще пять телефонных номеров», вы должны сказать им, что система не является заменой Outlook. Если пользователь хочет сохранить больше телефонных номеров, чем вы указали, поместите его в столбец «Примечание». С адресами вам нужно знать, сколько адресов вам нужно хранить. Если их больше двух (или даже одного), я бы порекомендовал поместить их в отдельную таблицу.
Гибкость связана 1: 1 со сложностью. Управление неограниченным количеством адресов электронной почты, телефонных номеров и адресов является более гибким, но добавляет больше работы в пользовательском интерфейсе. Большинство проектов захвата контактов на самом деле не нужно собирать десятки адресов электронной почты для данного контакта, и, следовательно, полностью нормализованный дизайн просто добавляет больше работы, чем необходимо.