Одни и те же данные из разных объектов в базе данных - Best Practice - Пример телефонных номеров

Довольно простой вопрос: если бы у меня была система, которая бы работала с персоналом, клиентами и поставщиками, у каждого из которых было несколько возможных телефонных номеров, как бы вы сохранили эти номера в хорошем нормированном виде? Я немного подумал, и логичный путь не выпрыгивает из меня.

3 ответа

Решение

В большинстве случаев.,,

  • "Персонал" всегда описывает людей.
  • Некоторые клиенты люди.
  • Некоторые клиенты являются предприятиями (организациями).
  • "Поставщиками" обычно являются (всегда?) Организации.
  • Персонал также может быть клиентом.
  • Поставщики также могут быть клиентами.

Существуют серьезные проблемы с наличием отдельных таблиц телефонных номеров сотрудников, телефонных номеров поставщиков и телефонных номеров клиентов.

  • Сотрудники могут быть клиентами. Если телефонный номер персонала меняется, необходимо ли обновлять номер телефона клиента? Как узнать, какой из них обновить?
  • Поставщики могут быть клиентами. Если номер телефона поставщика изменяется, необходимо ли также обновлять номер телефона клиента? Как узнать, какой из них обновить?
  • Вы должны дублировать и поддерживать без ошибок ограничения для телефонных номеров в каждой таблице, в которой хранятся телефонные номера.
  • Те же проблемы возникают при изменении номера телефона клиента. Теперь вам нужно проверить, нужно ли обновлять номера телефонов сотрудников и поставщиков.
  • Чтобы ответить на вопрос "Чей номер телефона 123-456-7890?", Вам нужно посмотреть в "n" разных таблицах, где "n" - это количество разных "видов" партий, с которыми вы имеете дело. В дополнение к персоналу, клиентам и поставщикам, подумайте о "телефонах подрядчика", "телефонах потенциальных клиентов" и т. Д.

Вам необходимо реализовать схему супертипа / подтипа. (Код PostgreSQL, строго не проверенный.)

create table parties (
    party_id integer not null unique,
    party_type char(1) check (party_type in ('I', 'O')),
    party_name varchar(10) not null unique,
    primary key (party_id, party_type)
);

insert into parties values (1,'I', 'Mike');
insert into parties values (2,'I', 'Sherry');
insert into parties values (3,'O', 'Vandelay');

-- For "persons", a subtype of "parties"
create table person_st (
    party_id integer not null unique,
    party_type char(1) not null default 'I' check (party_type = 'I'),
    height_inches integer not null check (height_inches between 24 and 108),
    primary key (party_id),
    foreign key (party_id, party_type) references parties (party_id, party_type) on delete cascade
);

insert into person_st values (1, 'I', 72);
insert into person_st values (2, 'I', 60);

-- For "organizations", a subtype of "parties"
create table organization_st (
    party_id integer not null unique,
    party_type CHAR(1) not null default 'O' check (party_type = 'O'),
    ein CHAR(10), -- In US, federal Employer Identification Number
    primary key (party_id),
    foreign key (party_id, party_type) references parties (party_id, party_type) on delete cascade
);

insert into organization_st values (3, 'O', '00-0000000');

create table phones (
    party_id integer references parties (party_id) on delete cascade,
    -- Whatever you prefer to distinguish one kind of phone usage from another.
    -- I'll just use a simple 'phone_type' here, for work, home, emergency, 
    -- business, and mobile.
    phone_type char(1) not null default 'w' check 
        (phone_type in ('w', 'h', 'e', 'b', 'm')),
    -- Phone numbers in the USA are 10 chars. YMMV.
    phone_number char(10) not null check (phone_number ~ '[0-9]{10}'),
    primary key (party_id, phone_type)
);

insert into phones values (1, 'h', '0000000000');
insert into phones values (1, 'm', '0000000001');
insert into phones values (3, 'h', '0000000002');

-- Do what you need to do on your platform--triggers, rules, whatever--to make 
-- these views updatable. Client code uses the views, not the base tables.
-- In current versions of PostgreSQL, I think you'd create some "instead
-- of" rules.
--
create view people as
select t1.party_id, t1.party_name, t2.height_inches
from parties t1
inner join person_st t2 on (t1.party_id = t2.party_id);

create view organizations as 
select t1.party_id, t1.party_name, t2.ein
from parties t1
inner join organization_st t2 on (t1.party_id = t2.party_id);

create view phone_book as
select t1.party_id, t1.party_name, t2.phone_type, t2.phone_number
from parties t1
inner join phones t2 on (t1.party_id = t2.party_id);

Чтобы продвинуть это немного дальше, таблица для реализации "персонала" должна ссылаться на подтип человека, а не на супертип партии. Организации не могут быть в штате.

create table staff (
    party_id integer primary key references person_st (party_id) on delete cascade,
    employee_number char(10) not null unique,
    first_hire_date date not null default CURRENT_DATE
);

Если поставщиками могут быть только организации, а не отдельные лица, то таблица, в которой реализованы поставщики, будет аналогичным образом ссылаться на подтип организаций.

Для большинства компаний заказчиком может быть либо физическое лицо, либо организация, поэтому таблица, в которой реализованы клиенты, должна ссылаться на супертип.

create table customers (
    party_id integer primary key references parties (party_id) on delete cascade
    -- Other attributes of customers
);

Я думаю, что решение должно основываться на практической оценке того, насколько важна эта контактная информация, как часто она меняется и насколько часто могут совпадать люди разных типов с телефонными номерами.

Если контактная информация нестабильна и / или действительно имеет центральное значение для приложения, то, скорее всего, будет лучше нормализация. Это будет означать наличие таблицы PHONE_NUMBER, на которую могут указывать ваши различные таблицы CUSTOMER, SUPPLIER, EMPLOYEE (и т. Д.) Или, что более вероятно, с каким-либо трехсторонним пересечением между типом контакта, контактным лицом (клиент / поставщик / сотрудник) и контактная точка (телефон). Таким образом, домашний телефон сотрудника может быть основным деловым номером их клиентов, и если он меняется, он меняется один раз для каждого использования этой контактной точки.

С другой стороны, если вы храните телефонные номера, и вы не используете их и, вероятно, не будете их обслуживать, то тратите много времени и усилий на моделирование и внедрение этой сложности в свою базу данных, и вы выиграете ". это того не стоит, и вы можете сделать хорошие, старомодные столбцы Phone1, Phone2, Phone3,... на CUSTOMER, SUPPLIER, EMPLOYEE или что у вас есть. Это плохой дизайн базы данных, но это хорошая практика разработки системы, поскольку она применяет правило 80/20 для определения приоритетов проекта.

Итак, подведем итог: если данные имеют значение, сделайте это правильно, если данные на самом деле не имеют значения, просто добавьте их - или еще лучше, оставьте их вообще.

Ответ Mike Sherrill 'Cat Recall' работает на MariaDB с одним единственным изменением: "~" должно стать "LIKE".

Вот его пример, проверенный на MariaDB. Я также внес изменения, о которых здесь просили, в отношении типов, описываемых с использованием слов, а не одиночных символов.

create table parties (
    party_id integer not null unique,
    party_type varchar(20) not null check (party_type in ('individual', 'organization')),
    party_name varchar(50) not null unique,
    primary key (party_id, party_type)
);

insert into parties values (1,'individual', 'Mike');
insert into parties values (2,'individual', 'Sherry');
insert into parties values (3,'organization', 'Vandelay');

-- For "persons", a subtype of "parties"
create table person_st (
    party_id integer not null unique,
    party_type varchar(20) not null default 'individual' check (party_type = 'individual'),
    height_inches integer not null check (height_inches between 24 and 108),
    primary key (party_id),
    foreign key (party_id, party_type) references parties (party_id, party_type) on delete cascade
);

insert into person_st values (1, 'individual', 72);
insert into person_st values (2, 'individual', 60);

-- For "organizations", a subtype of "parties"
create table organization_st (
    party_id integer not null unique,
    party_type varchar(20) not null default 'organization' check (party_type = 'organization'),
    ein CHAR(10), -- In US, federal Employer Identification Number
    primary key (party_id),
    foreign key (party_id, party_type) references parties (party_id, party_type) on delete cascade
);

insert into organization_st values (3, 'organization', '00-0000000');

create table phones (
    party_id integer references parties (party_id) on delete cascade,
    -- Whatever you prefer to distinguish one kind of phone usage from another.
    -- I'll just use a simple 'phone_type' here, for work, home, emergency,
    -- business, and mobile.
    phone_type varchar(10) not null default 'work' check
        (phone_type in ('work', 'home', 'emergency', 'business', 'mobile')),
    -- Phone numbers in the USA are 10 chars. YMMV.
    phone_number char(10) not null check (phone_number like '[0-9]{10}'),
    primary key (party_id, phone_type)
);

insert into phones values (1, 'home', '0000000000');
insert into phones values (1, 'mobile', '0000000001');
insert into phones values (3, 'home', '0000000002');

-- Do what you need to do on your platform--triggers, rules, whatever--to make
-- these views updatable. Client code uses the views, not the base tables.
-- Inserting and Updating with Views - MariaDB Knowledge Base https://mariadb.com/kb/en/library/inserting-and-updating-with-views/
--
create view people as
select t1.party_id, t1.party_name, t2.height_inches
from parties t1
inner join person_st t2 on (t1.party_id = t2.party_id);

create view organizations as
select t1.party_id, t1.party_name, t2.ein
from parties t1
inner join organization_st t2 on (t1.party_id = t2.party_id);

create view phone_book as
select t1.party_id, t1.party_name, t2.phone_type, t2.phone_number
from parties t1
inner join phones t2 on (t1.party_id = t2.party_id);

Самый простой способ, вероятно, лучший. Даже если у всех сотрудников, клиентов или поставщиков есть место для телефона, мобильного телефона и номера факса, лучше всего просто поместить эти поля в каждую таблицу.

Но чем больше таких полей у вас есть, тем больше вы должны рассматривать своего рода "наследование" или централизацию. Если есть другая контактная информация, а также несколько телефонных номеров, вы можете использовать эти общие значения в централизованной таблице " Контакты". Поля, относящиеся к типу "Клиент", "Поставщик" и т. Д., Будут находиться в отдельных таблицах. Например, таблица Customer будет иметь внешний ключ ContactID для контактов.

Другие вопросы по тегам