INSERT Employees (ID,Name,Birthday,Email,PositionID,DepartmentID)VALUES

(1000,N'Иванов И.И.','19550219','[email protected]',2,1),

(1001,N'Петров П.П.','19831203','[email protected]',3,3),

(1002,N'Сидоров С.С.','19760607','[email protected]',1,2),

(1003,N'Андреев А.А.','19820417','[email protected]',4,3)

Немного про индексы, создаваемые при создании ограничений PRIMARY KEY и UNIQUE


Как можно увидеть на скриншоте выше, при создании ограничений PRIMARY KEY и UNIQUE автоматически создались индексы с такими же названиями (PK_Employees и UQ_Employees_Email). По умолчанию индекс для первичного ключа создается как CLUSTERED, а для всех остальных индексов как NONCLUSTERED. Стоит сказать, что понятие кластерного индекса есть не во всех СУБД. Таблица может иметь только один кластерный (CLUSTERED) индекс. CLUSTERED – означает, что записи таблицы будут сортироваться по этому индексу, так же можно сказать, что этот индекс имеет непосредственный доступ ко всем данным таблицы. Это так сказать главный индекс таблицы. Если сказать еще грубее, то это индекс, прикрученный к таблице. Кластерный индекс – это очень мощное средство, которое может помочь при оптимизации запросов, пока просто запомним это. Если мы хотим сказать, чтобы кластерный индекс использовался не в первичном ключе, а для другого индекса, то при создании первичного ключа мы должны указать опцию NONCLUSTERED:


ALTER TABLE имя_таблицы ADD CONSTRAINT имя_ограничения

PRIMARY KEY NONCLUSTERED(поле1,поле2,…)


Для примера сделаем индекс ограничения PK_Employees некластерным, а индекс ограничения UQ_Employees_Email кластерным. Первым делом удалим данные ограничения:


ALTER TABLE Employees DROP CONSTRAINT PK_Employees

ALTER TABLE Employees DROP CONSTRAINT UQ_Employees_Email


А теперь создадим их с опциями CLUSTERED и NONCLUSTERED:


ALTER TABLE Employees ADD CONSTRAINT PK_Employees PRIMARY KEY NONCLUSTERED (ID)

ALTER TABLE Employees ADD CONSTRAINT UQ_Employees_Email UNIQUE CLUSTERED (Email)


Теперь, выполнив выборку из таблицы Employees, мы увидим, что записи отсортировались по кластерному индексу UQ_Employees_Email:


SELECT * FROM Employees

ID Name Birthday Email PositionID DepartmentID HireDate
Андреев А.А. 1982-04-17 [email protected] 2015-04-08
Иванов И.И. 1955-02-19 [email protected] 2015-04-08
Петров П.П. 1983-12-03 [email protected] 2015-04-08
Сидоров С.С. 1976-06-07 [email protected] 2015-04-08


До этого, когда кластерным индексом был индекс PK_Employees, записи по умолчанию сортировались по полю ID.

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

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

Кластерный индекс выгодно применять к полям, по которым выборка идет наиболее часто.

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

Подытожим


На данном этапе мы познакомились со всеми видами ограничений, в их самом простом виде, которые создаются командой вида «ALTER TABLE имя_таблицы ADD CONSTRAINT имя_ограничения …»:

· PRIMARY KEY – первичный ключ;

· FOREIGN KEY – настройка связей и контроль ссылочной целостности данных;

· UNIQUE – позволяет создать уникальность;

· CHECK – позволяет осуществлять корректность введенных данных;

· DEFAULT – позволяет задать значение по умолчанию;

· Так же стоит отметить, что все ограничения можно удалить, используя команду «ALTER TABLEимя_таблицы DROP CONSTRAINT имя_ограничения».


Так же мы частично затронули тему индексов и разобрали понятие кластерный (CLUSTERED) и некластерный (NONCLUSTERED) индекс.

Создание самостоятельных индексов


Под самостоятельностью я здесь имеется в виду индексы, которые создаются не для ограничения PRIMARY KEY или UNIQUE.

Индексы по полю или полям можно создавать следующей командой:


CREATE INDEX IDX_Employees_Name ON Employees(Name)


Так же здесь можно указать опции CLUSTERED, NONCLUSTERED, UNIQUE, а так же можно указать направление сортировки каждого отдельного поля ASC (по умолчанию) или DESC:


CREATE UNIQUE NONCLUSTERED INDEX UQ_Employees_EmailDesc ON Employees(Email DESC)


При создании некластерного индекса опцию NONCLUSTERED можно отпустить, т.к. она подразумевается по умолчанию, здесь она показана просто, чтобы указать позицию опции CLUSTERED или NONCLUSTERED в команде.

Удалить индекс можно следующей командой:


DROP INDEX IDX_Employees_Name ON Employees


Простые индексы так же, как и ограничения, можно создать в контексте команды CREATE TABLE.

Для примера снова удалим таблицу:


DROP TABLE Employees


И пересоздадим ее со всеми созданными ограничениями и индексами одной командой CREATE TABLE:


CREATE TABLE Employees(

ID int NOT NULL,

Name nvarchar(30),

Birthday date,

Email nvarchar(30),

PositionID int,

DepartmentID int,

HireDate date NOT NULL CONSTRAINT DF_Employees_HireDate DEFAULT SYSDATETIME(),

ManagerID int,

CONSTRAINT PK_Employees PRIMARY KEY (ID),

CONSTRAINT FK_Employees_DepartmentID FOREIGN KEY(DepartmentID) REFERENCES Departments(ID),

CONSTRAINT FK_Employees_PositionID FOREIGN KEY(PositionID) REFERENCES Positions(ID),

CONSTRAINT FK_Employees_ManagerID FOREIGN KEY (ManagerID) REFERENCES Employees(ID),

CONSTRAINT UQ_Employees_Email UNIQUE(Email),

CONSTRAINT CK_Employees_ID CHECK(ID BETWEEN 1000 AND 1999),

INDEX IDX_Employees_Name(Name)

)


Напоследок вставим в таблицу наших сотрудников:


Наши рекомендации