Почему не стоит передавать объекты в качестве моделей в MVC? - PullRequest
14 голосов
/ 25 февраля 2010

Мы разрабатываем довольно большое приложение с MVC 2 RC2, и мы получили некоторые отзывы о том, как мы используем Lazy Loading Entity Framework.

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

Можете ли вы помочь нам понять эту проблему дизайна?

Спасибо!

Ответы [ 4 ]

22 голосов
/ 25 февраля 2010

Основной проблемой здесь является сцепление. Идея модели , обозначающей букву «M» в «MVC», заключается в том, что она не имеет внешних зависимостей. Это «ядро» вашего приложения. Дерево зависимостей хорошо спроектированной архитектуры приложения должно выглядеть примерно так:

                       +---------------------------------------> Views
                       |                                           |
                       |                                           |
                       |                                           v
                  Controllers ----+-> Model Transformer -----> View Model
                       |           \         |
                       |            \        |
                       v             \       v
Data Access <---- Persistence --------> Domain Model
                       |             /
                       |            /
                       v           /
                     Mapper ------+

Теперь я понимаю, что не совсем убедительно просто сказать «вот архитектура, это то, что вы должны использовать», поэтому позвольте мне объяснить, что здесь происходит:

  1. Контроллер получает запрос.
  2. Контроллер вызывает какой-то постоянный уровень (то есть хранилище).
  3. Слой постоянства извлекает данные, а затем использует сопоставитель для сопоставления с моделью домена.
  4. Контроллер использует преобразователь для преобразования модели предметной области в модель представления.
  5. Контроллер выбирает необходимый вид и применяет к нему модель вида.

Итак, почему это хорошо?

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

  • Постоянный уровень возвращает экземпляры модели домена . Это означает, что его можно смоделировать как полностью абстрактный, независимый от платформы интерфейс. Компонент, который должен использовать постоянный уровень (например, контроллер), не принимает никаких дополнительных зависимостей. Это идеально подходит для внедрения зависимостей в слой постоянства и, опять же, в тестируемость. Комбинация постоянства, доступа к данным и картографа может существовать в своей собственной сборке. В более крупных проектах вы даже сможете дополнительно отделить маппер и заставить его работать с общим набором записей.

  • Контроллер имеет только две нижестоящие зависимости - модель домена и уровень персистентности. Модель должна меняться редко, так как это ваша бизнес-модель, и поскольку уровень постоянства является абстрактным, контроллер почти никогда не нужно менять (кроме добавления новых действий).

  • Представления зависят от отдельной модели пользовательского интерфейса. Это изолирует их от изменений в доменной модели. Это означает, что если ваша бизнес-логика меняется, вам не нужно менять каждое представление в вашем проекте. Это позволяет представлениям быть "тупыми", как и должно быть представлениями - они не намного больше, чем заполнители для данных представления. Это также означает, что должно быть просто воссоздать представление с использованием пользовательского интерфейса другого типа, то есть приложения интеллектуального клиента, или переключиться на другой механизм представления (Spark, NHaml и т. Д.)


Теперь, при использовании O / R Mappers, таких как Linq to SQL или Entity Framework, очень заманчиво рассматривать классы, которые они генерируют, как модель вашего домена. Это, конечно, выглядит как модель домена, но это не так. Почему?

  • Классы сущностей привязаны к вашей реляционной модели, которая со временем может и будет значительно отличаться от вашей доменной модели;

  • Классы сущностей глупы. Трудно поддерживать какие-либо сложные сценарии проверки или интегрировать какие-либо бизнес-правила. Это называется модель анемичной области .

  • Классы сущностей имеют скрытые зависимости. Хотя они могут казаться обычными POCO, на самом деле они могут иметь скрытые ссылки на базу данных (т.е. ленивая загрузка ассоциаций). Это может привести к возникновению проблем, связанных с базой данных, к логике представления, где вы меньше всего сможете правильно проанализировать происходящее и отладить.

  • Но самое главное, «модель предметной области» больше не является независимой. Она не может существовать вне какой-либо сборки, имеющей логику доступа к данным. Ну, это как-то может, есть способы пойти на это, если вы действительно работаете над этим, но большинство людей так не делают, и даже если вы это сделаете, вы обнаружите, что фактический дизайн модели предметной области ограничены вашей реляционной моделью и, в частности, поведением EF. Суть в том, что если вы решите изменить свою модель персистентности, вы сломаете модель домена, и ваша модель домена станет основой для всего всего остального в вашем приложении.

Классы Entity Framework являются , а не моделью домена. Они являются частью реляционной модели данных и имеют те же или похожие имена, которые вы можете дать классам в модели предметной области. Но это разные миры с точки зрения управления зависимостями. Использование классов, сгенерированных из инструмента ORM, в качестве модели вашего домена может привести только к чрезвычайно хрупкой архитектуре / дизайну; Каждое изменение, которое вы вносите в почти любую часть приложения, будет иметь множество предсказуемых и непредсказуемых каскадных эффектов.

Многие люди считают, что вам не нужна сплоченная, независимая модель предметной области. Обычно оправданием является то, что (а) это небольшой проект, и / или (б) их модель предметной области на самом деле не имеет никакого поведения. Но небольшие проекты становятся большими, и бизнес-правила становятся (намного) более сложными, и анемичная или несуществующая модель предметной области - это не то, что вы можете просто реорганизовать.

Это на самом деле самая коварная черта дизайна сущностей как модель; кажется, на некоторое время нормально работает . Вы не узнаете, какая это большая ошибка, до тех пор, пока через год или два вы не утонете в сообщениях о дефектах и ​​запросах на изменения, отчаянно пытаясь собрать реальную модель предметной области.

2 голосов
/ 25 февраля 2010

Проблема в том, что ваш пользовательский интерфейс более привязан к вашей сущности, чем необходимо.

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

2 голосов
/ 25 февраля 2010

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

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

Кроме того, вы позволяете своей зависимости от EF переходить к представлению и проникать в ваше приложение, где вы должны стараться быть как можно слабее связанным с этой зависимостью.

2 голосов
/ 25 февраля 2010

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

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...