Занятие 5-3. определение связи «многие ко многим»
Содержание:
- Пример рынка “один ко многим”
- Для чего все это нужно?
- Как определить связи между таблицами
- Создание и изменение отношений 1:N между сущностями
- Значения ON DELETE и ON UPDATE
- Один к одному (one-to-one)
- Работа с моделями со связью многие-ко-многим
- Отношение один ко многим
- Связь один-к-одному
- Добавление необходимых таблиц к представлению источника данных
Пример рынка “один ко многим”
Самым печально известным примером торговой платформы «один ко многим» была фиктивная торговля привели к довольно быстрой гибели Enron EOL.
Enron выступал в качестве контрагента по каждой транзакции, совершаемой на бирже.Это означает, что кредит Enron использовался для каждой транзакции.На нормальном рынке клиринговая палата гарантирует, что обе стороны сделки получат то, что им положено.На нерегулируемом или внебиржевом рынке существует риск контрагента.Этот тип риска возникает из-за незнания, сможет ли другая сторона выполнить свою сторону сделки.
Изначально у Enron была хорошая репутация и кредит, но вскоре начали образовываться трещины. Enron больше не могла откладывать дела до конца.Трейдеры, совершавшие сделки с Enron, также сбежали, оставив их без дохода, необходимого для поддержки своего неудачливого бизнеса в других сферах.
Хотя проект EOL и Enron потерпели неудачу, какое-то время он был успешным для Enron.В 2000 году объем торгов на платформе составил более 300 миллиардов долларов2.
#О
Для чего все это нужно?
Связи выполняют более важную роль, чем просто информация размещения данных по таблицам. Прежде всего они требуются разработчикам для поддержания целостности баз данных.
Правильно настроив связи, можно быть уверенным, что ничего не потеряется.
Представьте, что Вы решили удалить одну из групп в таблице учебной базы данных. Если бы связи не было, то для тех сотрудников, которые к ней были определены, остался идентификатор несуществующей группы. Связь не позволит удалить группу, пока она имеется во внешних ключах других таблиц. Для начала следовало определить сотрудников в другие имеющиеся или новые группы, а только затем удалить ненужную запись. Поэтому связи называют еще ограничениями.
- < Назад
- Вперёд >
Новые статьи:
-
Объединение таблиц – UNION
-
Соединение таблиц – операция JOIN и ее виды
-
Тест на знание основ SQL
Если материалы office-menu.ru Вам помогли, то поддержите, пожалуйста, проект, чтобы я мог развивать его дальше.
Как определить связи между таблицами
При создании связи между таблицами связанные поля не должны иметь одни и те же имена. Однако связанные поля должны иметь один и тот же тип данных, если только поле первичного ключа не является полем AutoNumber. Вы можете сопоставить поле AutoNumber с полем Number, только если свойство FieldSize обоих совпадающих полей совпадает. Например, можно сопоставить поле AutoNumber и поле Number, если свойство theFieldSizeproperty обоих полей имеет значение Long Integer. Даже если оба совпадающих поля являются числовыми полями, они должны иметь параметр sameFieldSizeproperty.
Как определить связи «один ко многим» или «один к одному»
Чтобы создать связь «один ко многим» или «один к одному», выполните следующие действия.
-
Закройте все таблицы. Нельзя создавать или изменять связи между открытыми таблицами.
-
В Access 2002 и Access 2003 выполните следующие действия.
- Нажмите F11, чтобы переключиться в окно базы данных.
- В меню Инструменты выберите Связи.
В Access 2007, Access 2010 или Access 2013 нажмите Связи в группе Показать/Скрыть на вкладке Инструменты базы данных.
-
Если вы еще не определили какие-либо связи в базе данных, автоматически отобразится диалоговое окно Показать таблицу. Если вы хотите добавить таблицы, которые нужно связать, но диалоговое окно Показать таблицу не отображается, нажмите Показать таблицу в меню Связи.
-
Дважды щелкните названия таблиц, которые вы хотите связать, а затем закройте диалоговое окно Показать таблицу. Чтобы создать связь между одной и той же таблицей, добавьте эту таблицу два раза.
-
Перетащите поле, которое вы хотите связать, из одной таблицы в связанное поле в другой таблице. Чтобы перетащить несколько полей, нажмите Ctrl, нажмите на каждое поле, а затем перетащите их.
В большинстве случаев вы перетаскиваете поле первичного ключа (это поле отображается жирным текстом) из одной таблицы в аналогичное поле (это поле часто имеет одно и то же имя), которое называется внешним ключом в другой таблице.
-
Откроется диалоговое окно Изменение связей. Убедитесь, что имена полей, отображаемые в двух столбцах, верны. Вы можете изменить имена, если это необходимо.
При необходимости установите параметры связей. Если у вас есть информация о конкретном элементе в диалоговом окне Изменение связей, нажмите кнопку со знаком вопроса, а затем нажмите на элемент. (Эти параметры будут подробно описаны ниже в этой статье.)
-
Нажмите кнопку Создать, чтобы создать связь.
-
Повторите шаги с 4 по 7 для каждой пары таблиц, которые вы хотите связать.
При закрытии диалогового окна Изменение связей Access спрашивает, хотите ли вы сохранить макет. Сохраняете ли вы макет или не сохраняете макет, созданные вами связи сохраняются в базе данных.
Примечание
Можно создавать связи не только в таблицах, но и в запросах. Однако целостность данных связывания не обеспечивается с помощью запросов.
Как определить связь «многие ко многим»
Чтобы создать связь «многие ко многим», выполните следующие действия.
-
Создайте две таблицы, которые будут иметь связь «многие ко многим».
-
Создайте третью таблицу. Это стыковочная таблица. В таблице соединения добавьте новые поля, которые имеют те же определения, что и основные ключевые поля из каждой таблицы, созданной в шаге 1. В связующей таблице основные ключевые поля функционируют как внешние ключи. Вы можете добавить другие поля в связующую таблицу, так же, как и в любую другую таблицу.
-
В связующей таблице установите первичный ключ, чтобы включить основные ключевые поля из двух других таблиц. Например, в связующей таблице «TitleAuthors» первичный ключ будет состоять из полей OrderID и ProductID.
Примечание
Чтобы создать первичный ключ, выполните следующие действия:
-
Откройте таблицу в Конструкторе.
-
Выберите поле или поля, которые вы хотите определить в качестве первичного ключа. Чтобы выбрать одно поле, нажмите на селектор строки для нужного поля. Чтобы выбрать несколько полей, удерживайте клавишу Ctrl, а затем нажмите селектор строки для каждого поля.
-
В Access 2002 или в Access 2003 нажмите на Первичный ключ на панели инструментов.
В Access 2007 нажмите на Первичный ключ в группе Инструменты на вкладке Дизайн.
Примечание
Если вы хотите, чтобы порядок полей в первичном ключе с несколькими полями отличался от порядка этих полей в таблице, нажмите Индексы на панели инструментов для отображения диалогового окна Indexes, а затем заново упорядочите имена полей для индекса с именем PrimaryKey.
-
-
Определите связь один-ко-многим между каждой основной и связующей таблицами.
Создание и изменение отношений 1:N между сущностями
Откройте обозреватель решений.
В разделе Компоненты раскройте узел Сущности, затем раскройте сущность, с которой требуется работать.
Выберите Отношения 1:N.
Чтобы изменить отношение или просмотреть сведения для отношения, выберите отношение и нажмите на панели инструментов «Действия» кнопку Другие действия, затем выберите Изменить.
— ИЛИ —
Чтобы добавить новое отношение, выберите Создать отношение «один ко многим».
Важно!
Если кнопка Создать отношение «один ко многим» не отображается на панели инструментов «Действия», то создать отношение 1:N для этой сущности невозможно.
Для нового отношения в разделе Определение отношения выберите в списке Связанная сущность сущность для связывания.
Примечание
При указании связанной сущности задается значение по умолчанию в поле Имя. Если изменить связанную сущность перед ее сохранением, соответственно изменится и значение поля Имя.
Выберите, будет ли это поле доступно для поиска или нет.
В разделе Поле поиска укажите значение для поля в поле Отображаемое имя.
Важно!
При указании значения Отображаемое имя задается значение по умолчанию в поле Имя
Если изменить Отображаемое имя поля поиска перед сохранением данных, значение в поле Имя не изменится. Поэтому необходимо ввести в поле Имя информативное значение перед сохранением данных.
В списке Требование поля выберите вариант, чтобы указать требования к данным для поля перед сохранением записи.
В разделе Элемент области переходов для основной сущности в списке Параметры отображения выберите вариант отображения связанных представлений для пользовательской метки.
В разделе Поведение отношений выберите в списке Тип отношений один из следующих вариантов.
Родительское. В родительском отношении между двумя сущностями любое действие, выполняемое над записью основной (родительской) сущности, также выполняется над всеми связанными с ней записями дочерних сущностей.
Ссылочное. При ссылочном отношении между двумя сущностями можно переходить к любым связанным записям, но действия, выполняемые над одной записью, не применяются к другим.
Ссылочное с ограниченным удалением. В ссылоном отношении с ограничением удаления можно переходить к любым связанным записям. Действия, выполняемые над родительской записью, не будут выполняться над дочерней, но пока она существует, удалить родительскую запись будет невозможно. Учтите, что запись нельзя удалить, если имеются связанные с ней записи.
Настраиваемое каскадное. В настраиваемом каскадном отношении между двумя сущностями выбирается поведение, связанное с каждым из наборов возможных действий.
Важно!
Если выбрать поведения для действий, совпадающие с поведениями для действий, связанными с другим Типом поведения, то при сохранении отношения значение Тип поведения будет автоматически установлено равным такому совпадающему типу.
Дополнительные сведения:
-
Выберите Сохранить и закрыть, чтобы закрыть форму Отношение.
-
Выполнив настройки, опубликуйте их:
-
Чтобы опубликовать настройки только для компонента, изменяемого в данный момент, на панели инструментов «Действия» выберите Опубликовать.
-
Чтобы опубликовать настройки для всех неопубликованных компонентов одновременно, на панели навигации или в области переходов выберите Сущности, затем на панели инструментов «Действия» выберите Опубликовать все настройки.
-
Примечание
- Настраиваемая сущность не может быть основной в каскадном отношении со связанной системной сущностью. Это означает, что между основной настраиваемой сущностью и связанной системной сущностью не может быть отношений с каким-либо из действий, установленным в «Передавать всем», «Передавать активным» или «Передавать владельцу».
- У новых отношений действие не может иметь значение Передавать всем, Передавать активным или Передавать владельцу, если связанная сущность в этом отношении уже является связанной сущностью в любом другом отношении, действие которого имеет значение Передавать всем, Передавать активным или Передавать владельцу. Это позволяет избежать создания отношений с несколькими родительскими сущностями.
- После каждого изменения элементов пользовательского интерфейса или внедрения скриптов формы для сущности необходима публикация изменений. Все изменения в схеме данных приложения, таких как настраиваемые сущности, связи или поля, применяются сразу.
- Если отношение является частью управляемого решения, разработчик решения может ограничить настройку отношения пользователями.
- Установка решения или публикация настроек может помешать нормальной работе системы. Рекомендуется запланировать импорт решения в оптимальный для пользователей период.
Значения ON DELETE и ON UPDATE
CASCADE — Каскадное удаление и редактирование. Эта настройка означает, что при удалении каталога, все товары из него тоже удалятся. При редактировании, если мы изменим id каталога, у товаров автоматически изменится поле «catalog_id».
RESTRICT — При этой настройке, если мы попытаемся удалить каталог, в котором есть товары, или изменить его id, база данных выдаст нам ошибку и удаление не состоится.
SET NULL — Из названия видно, что если исчезнет(удалится или изменится) каталог с таким id, то у товаров в поле «catalog_id» установится значение NULL
С этой настройкой нужно вести себя осторожно, потому что по умолчанию индексы «NOT NULL»
NO ACTION — Игнорируем удаление и редактирование каталога, и пусть в поле «catalog_id» будет несуществующий идентификатор, просто игнорируем это.
Один к одному (one-to-one)
Связь один к одному это когда одной записи в таблице отвечает только одна запись из другой таблицы. Чтобы продемонстрировать связь один к одному возьмем таблицу supplier:
и вынесем колонку full_address в отдельную таблицу. Таким образом в таблице supplier будет ссылка на таблицу address, в которой будут такие поля: address_id, coutry, city, street.
Не теряя времени сделаем нужные изменения.
alter table supplier modify column full_address int;
Так как поле full_address будет ссылкой на таблицу address сделаем его тип целочисленным.
Далее создадим саму таблицу address.
create table address(address_id int auto_increment primary key, country text, city text, street text);
Теперь укажем, что поле full_address будет внешним ключом:
alter table supplier add foreign key (full_address) references address(address_id);
Связь один к одному означает, что в таблице supplier будет уникальный идентификатор записи с таблицы address. Поэтому, нужно сделать поле full_address уникальным:
alter table supplier add foreign key (full_address) references address(address_id);
Теперь, каждый адрес будет относиться только к одному поставщику и все попытки добавить поставщику адрес другого поставщика будет приводить к ошибке.
Работа с моделями со связью многие-ко-многим
Последнее обновление: 31.10.2015
Простой вывод связанных данных со связью «многие-ко-многим» не представляет особого труда. Однако не совсем понятно, как делать создание и
редактирование подобных моделей. Продолжим начатый в прошлой теме пример со студентами и курсами и внесем в него функциональность редактирования
студентов.
В итоге форма редактирования у нас будет выглядеть примерно так:
Итак, добавим в контроллер HomeController следующее действие Edit:
public ActionResult Edit(int id = 0) { Student student = db.Students.Find(id); if (student == null) { return HttpNotFound(); } ViewBag.Courses = db.Courses.ToList(); return View(student); } public ActionResult Edit(Student student, int[] selectedCourses) { Student newStudent = db.Students.Find(student.Id); newStudent.Name = student.Name; newStudent.Surname = student.Surname; newStudent.Courses.Clear(); if (selectedCourses != null) { //получаем выбранные курсы foreach (var c in db.Courses.Where(co => selectedCourses.Contains(co.Id))) { newStudent.Courses.Add(c); } } db.Entry(newStudent).State = EntityState.Modified; db.SaveChanges(); return RedirectToAction("Index"); }
Действие Edit представляет два метода — для запроса get и для запроса post. Метод, обрабатывающий запрос get, стандартный — он передает
в представление редактируемую модель, а также список всех курсов через , чтобы мы могли
затем эти курсы вывести в представлении.
Метод post принимает полученные данные, только кроме модели Student сюда также передаются все выбранные курсы в виде массива id курсов. В
этот массив и будут помещаться все значения всех отмеченных на форме флажков. В самом методе мы устанавливаем новые значения свойств модели.
Затем нам надо установить в коллекции Courses у студента все отмеченные курсы. Для этого сначала удаляем у студента все курсы, затем проходим по всем курсам из базы данных, и если они были
отмечены на форме и отсутствуют в списке, то добавляем их. Неотмеченные курсы, если они раньше имелись у студента, таким образом, будут удалены. В итоге информация в базе данных будет соотвествующим образом обновлена.
И напоследок добавим само представление Edit.cshtml:
@using TeamMvc4.Models @model Student @{ ViewBag.Title = "Edit"; } @using (Html.BeginForm()) { <fieldset> <legend>Студент</legend> @Html.HiddenFor(model => model.Id) <div class="editor-label"><b>Имя</b></div> <div class="editor-field"> @Html.EditorFor(model => model.Name) </div> <div class="editor-label"><b>Фамилия</b></div> <div class="editor-field"> @Html.EditorFor(model => model.Surname) </div> <div class="editor-label"><b>Курсы</b></div> @{ List<Course> courses = ViewBag.Courses; foreach (Course c in courses) { <input type="checkbox" name="selectedCourses" value="@c.Id" @(Model.Courses.Contains(c) ? "checked=\"checked\"" : "") />@c.Name <br /> } } <p> <input type="submit" value="Сохранить" /> </p> </fieldset> }
Ну а сохранение модели будет во многом идентично редактированию.
НазадВперед
Отношение один ко многим
Последнее обновление: 17.11.2020
Связь один-ко-многим (one-to-many) представляет ситуацию, когда одна модель хранит ссылку на один объект другой модели, а вторая
модель может ссылаться на коллекцию объектов первой модели. Например, в одной компании может работать несколько сотрудников, а каждый сотрудник в
свою очередь может официально работать только в одной компании:
public class Company { public int Id { get; set; } public string Name { get; set; } public List<User> Users { get; set; } = new List<User>(); // сотрудники компании } public class User { public int Id { get; set; } public string Name { get; set; } public int CompanyId { get; set; } public Company Company { get; set; } // компания пользователя } public class ApplicationContext : DbContext { public DbSet<Company> Companies { get; set; } public DbSet<User> Users { get; set; } protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder) { optionsBuilder.UseSqlServer(@"Server=(localdb)\mssqllocaldb;Database=relationsdb;Trusted_Connection=True;"); } }
Добавление данных:
using (ApplicationContext db = new ApplicationContext()) { // пересоздадим базу данных db.Database.EnsureDeleted(); db.Database.EnsureCreated(); // создание и добавление моделей Company microsoft = new Company { Name = "Microsoft" }; Company google = new Company { Name = "Google" }; db.Companies.AddRange(microsoft, google); User tom = new User { Name = "Tom", Company = microsoft }; User bob = new User { Name = "Bob", Company = microsoft }; User alice = new User { Name = "Alice", Company = google }; db.Users.AddRange(tom, bob, alice); db.SaveChanges(); }
Получение данных:
using (ApplicationContext db = new ApplicationContext()) { // вывод пользователей var users = db.Users.Include(u => u.Company).ToList(); foreach (User user in users) Console.WriteLine($"{user.Name} - {user.Company?.Name}"); // вывод компаний var companies = db.Companies.Include(c => c.Users).ToList(); foreach (Company comp in companies) { Console.WriteLine($"\n Компания: {comp.Name}"); foreach (User user in comp.Users) { Console.WriteLine($"{user.Name}"); } } }
Редактирование:
using (ApplicationContext db = new ApplicationContext()) { // изменение имени пользователя User user1 = db.Users.FirstOrDefault(p => p.Name == "Tom"); if (user1!=null) { user1.Name = "Tomek"; db.SaveChanges(); } // изменение названия компании Company comp = db.Companies.FirstOrDefault(p => p.Name == "Google"); if (comp != null) { comp.Name = "Alphabet"; db.SaveChanges(); } // смена компании сотрудника User user2 = db.Users.FirstOrDefault(p => p.Name == "Bob"); if (user2 != null && comp!=null) { user2.Company = comp; db.SaveChanges(); } }
Удаление:
using (ApplicationContext db = new ApplicationContext()) { User user1 = db.Users.FirstOrDefault(p => p.Name == "Bob"); if (user1!=null) { db.Users.Remove(user1); db.SaveChanges(); } Company comp = db.Companies.FirstOrDefault(); if (comp != null) { db.Companies.Remove(comp); db.SaveChanges(); } }
Следует учитывать, что если зависимая сущность (в данном случае User) требует обязательного наличия главной сущности (в данном случае Company),
то на уровне базы данных при удалении главной сущности с помощью каскадного удаления будут удалены и связанные с ней зависимые сущности. Так,
в данном случае для объекта User установлено обязательное наличие объекта Company:
public int CompanyId { get; set; } public Company Company { get; set; } // компания пользователя
Соответственно при удалении компании будут удалены и все связанные с ней пользователи.
НазадВперед
Связь один-к-одному
Последнее обновление: 31.10.2015
Строго говоря в Entity Framework нет как таковой связи один-к-одному, так как ожидается, что обработчик будет использовать связь
один-ко-многим. Но все же нередко возникает потребность в наличие подобной связи между объектами в приложении, и в Entity Framework мы можем настроить
данный тип отношений.
Рассмотрим стандартный пример подобных отношений: есть класс пользователя User, который хранит логин и пароль, то есть данные учетных записей. А все данные профиля, такие
как имя, возраст и так далее, выделяются в класс профиля UserProfile.
using System.ComponentModel.DataAnnotations; using System.ComponentModel.DataAnnotations.Schema; //................................. public class User { public int Id { get; set; } public string Login { get; set; } public string Password { get; set; } public UserProfile Profile { get; set; } } public class UserProfile { public int Id { get; set; } public string Name { get; set; } public int Age { get; set; } public User User { get; set; } }
В этой связи между классами класс UserProfile является дочерним или подчиненным по отношению к классу User. И чтобы установить связь одни к одному,
у подчиненного класса устанавливается свойство идентификатора, которое называется также, как и идентификатор в основном классе. То есть в классе User свойство называется
Id, то и в UserProfile также свойство называется Id. Если бы в классе User свойство называлось бы UserId, то такое же название должно было быть и в UserProfile.
И в классе UserProfile над этим свойством Id устанавливаются два атрибута: , который показывает, то это первичный ключ, и
, который показывает, что это также и внешний ключ. Причем внешний ключ к таблице объектов User.
Соответственно классы User и UserProfile имеют ссылки друг на друга.
В классе контекста определяются свойства для взаимодействия с талицами в бд:
public class UserContext : DbContext { public DbSet<User> Users { get; set; } public DbSet<UserProfile> UserProfiles { get; set; } }
Для этих классов контекст данных будет создавать следующую таблицу UserProfiles:
CREATE TABLE . ( INT NOT NULL, NVARCHAR (MAX) NULL, INT NOT NULL, CONSTRAINT PRIMARY KEY CLUSTERED ( ASC), CONSTRAINT FOREIGN KEY () REFERENCES . () );
Посмотрим, как работать с моделями с такой связью. Добавление и получение:
using(UserContext db = new UserContext()) { User user1 = new User { Login = "login1", Password = "pass1234" }; User user2 = new User { Login = "login2", Password = "5678word2" }; db.Users.AddRange(new List<User> { user1, user2 }); db.SaveChanges(); UserProfile profile1 = new UserProfile { Id = user1.Id, Age = 22, Name = "Tom" }; UserProfile profile2 = new UserProfile { Id = user2.Id, Age = 27, Name = "Alice" }; db.UserProfiles.AddRange(new List<UserProfile> { profile1, profile2 }); db.SaveChanges(); foreach(User user in db.Users.Include("Profile").ToList()) Console.WriteLine("Name: {0} Age: {1} Login: {2} Password: {3}", user.Profile.Name, user.Profile.Age, user.Login, user.Password); }
Редактирование:
using (UserContext db = new UserContext()) { User user1 = db.Users.FirstOrDefault(); if(user1!=null) { user1.Password = "dsfvbggg"; db.Entry(user1).State = EntityState.Modified; db.SaveChanges(); } UserProfile profile2 = db.UserProfiles.FirstOrDefault(p => p.User.Login == "login2"); if(profile2!=null) { profile2.Name = "Alice II"; db.Entry(profile2).State = EntityState.Modified; db.SaveChanges(); } }
При удалении надо учитывать следующее: так как объект UserProfile требует наличие объекта User и зависит от этого объекта, то при удалении связанного
объекта User надо будет удалить и связанный с ним объект UserProfile. Поскольку по молчанию у нас не предусмотрено каскадное даление при данной связи.
Если же будет удален объект UserProfile, на объект User это никак не повлияет:
using (UserContext db = new UserContext()) { User user1 = db.Users.Include("Profile").FirstOrDefault(); if(user1!=null) { db.UserProfiles.Remove(user1.Profile); db.Users.Remove(user1); db.SaveChanges(); } UserProfile profile2 = db.UserProfiles.FirstOrDefault(p => p.User.Login == "login2"); if(profile2!=null) { db.UserProfiles.Remove(profile2); db.SaveChanges(); } }
НазадВперед
Добавление необходимых таблиц к представлению источника данных
Откройте конструктор представлений источника данных для представления источников данных Adventure Works DW 2012 .
Щелкните правой кнопкой мыши панель Организатор диаграмм , выберите команду Создать диаграмму и укажите Причина заказа через Интернет в качестве имени созданной диаграммы.
Перетащите таблицу InternetSales с панели Таблицы на панель Диаграмма .
Щелкните правой кнопкой мыши панель Диаграмма и выберите команду Добавить или удалить таблицы.
В диалоговом окне Добавление или удаление таблиц добавьте в список Включенные объекты таблицы DimSalesReason и FactInternetSalesReason , а затем нажмите кнопку ОК.
Обратите внимание, что связи «первичный-внешний ключ» между задействованными таблицами устанавливаются автоматически, поскольку эти связи определяются в базовой реляционной базе данных. Если эти связи не определены в базовой реляционной базе данных, их следует определить в представлении источника данных.
В меню Формат выберите команду Автоматический макет и щелкните значок Диаграмма.
В окне свойств измените свойство FriendlyName таблицы DimSalesReason на SalesReason, затем измените свойство FriendlyName таблицы FactInternetSalesReason на InternetSalesReason.
На панели Таблицы разверните узел InternetSalesReason (dbo.FactInternetSalesReason), щелкните столбец SalesOrderNumber и просмотрите в окне свойств свойство DataType для этого столбца данных.
Обратите внимание, что в качестве типа данных для столбца SalesOrderNumber указан тип данных string.
Просмотрите типы данных для других столбцов таблицы FactInternetSalesReason .
Обратите внимание, что для остальных двух столбцов этой таблицы указаны числовые типы данных.
На панели Таблицы щелкните правой кнопкой мыши таблицу InternetSalesReason (dbo.FactInternetSalesReason) и выберите пункт Просмотр данных.
Обратите внимание, что для каждого номера строки каждого заказа значение ключа указывает причину покупки данной позиции линии, как показано на следующем рисунке.