MVC3 - Вставка с помощью ViewModel - Ссылка на объект не установлена в экземпляр объекта
У меня есть две модели, как показано ниже, и я пытаюсь вставить одну из них в базу данных из одного представления. Я создал модель представления в попытке сделать это.
public class Blog
{
public int BlogID { get; set; }
public string Title { get; set; }
public DateTime CreatedOn { get; set; }
public virtual User User { get; set; }
public virtual ICollection<BlogPost> Posts { get; set; }
}
public class BlogPost
{
public int PostID { get; set; }
public string Body { get; set; }
public DateTime CreatedOn { get; set; }
public int UserID { get; set; }
public virtual User User { get; set; }
}
public class BlogViewModel
{
public Blog Blog { get; set; }
public BlogPost BlogPost { get; set; }
}
Используя модель представления, я отправляю на контроллер создания:
[HttpPost]
public ActionResult Create(BlogViewModel blog)
{
if (ModelState.IsValid)
{
User user = unit.UserRepository.GetUser();
blog.Blog.User = user;
blog.Blog.CreatedOn = DateTime.Now;
unit.BlogRepository.Insert(blog.Blog);
unit.BlogPostRepository.Insert(blog.BlogPost);
unit.Save();
return RedirectToAction("Index");
}
return View(blog);
}
Это предотвращает ошибку
Ссылка на объект не установлена в экземпляр объекта.
в строке blog.Blog.User = user
.
Любые мысли о том, что я делаю неправильно?
ИЗМЕНИТЬ
Я проверил данные POST, и все было там и исправлено, но все было размещено как Blog.Title =
и BlogPost.Body =
, поэтому модель представления в контроллере ничего не получала. Если я изменил действие actionresult контроллера на:
public ActionResult Create(Blog blog, BlogPost post)
Тогда все работает. Так почему же данные не отправляются в формате viewmodel? Я использую явное взаимодействие на основе модели представления между представлениями и контроллером.
@model Test.Models.BlogViewModel
@using (Html.BeginForm())
{
@Html.ValidationSummary(true)
<fieldset>
<legend>Blog</legend>
<div class="editor-label">
@Html.LabelFor(model => model.Blog.Title)
</div>
<div class="editor-field">
@Html.EditorFor(model => model.Blog.Title)
@Html.ValidationMessageFor(model => model.Blog.Title)
</div>
<div class="editor-label">
@Html.LabelFor(model => model.BlogPost.Body)
</div>
<div class="editor-field">
@Html.EditorFor(model => model.BlogPost.Body)
@Html.ValidationMessageFor(model => model.BlogPost.Body)
</div>
<p>
<input type="submit" value="Create" />
</p>
</fieldset>
}
<div>
@Html.ActionLink("Back to List", "Index")
</div>
Ответы
Ответ 1
Просто переименуйте аргумент action:
public ActionResult Create(BlogViewModel viewModel)
Существует конфликт, потому что ваш аргумент действия называется blog
, и все же ваша модель представления (BlogViewModel
) имеет свойство, называемое blog
. Проблема в том, что связующее устройство по умолчанию больше не знает, что делать в такой ситуации.
О, и если вам абсолютно необходимо, чтобы ваш аргумент действия назывался blog
, вы также можете переименовать свойство blog
в своей модели представления.
Ответ 2
Я раньше сталкивался с подобными ситуациями. Это происходит, когда данные, которые вы отправляете с вашего представления, не в формате, этот механизм привязки модели MVC по умолчанию может понимать или может отображать ваш объект модели просмотра. Попробуйте использовать "FireBug" и посмотрите, как данные передаются через POST. А затем посмотрите, могут ли эти данные сопоставляться с определением вашей модели представления.
Еще одна вещь: используете ли вы jQuery для передачи данных или используете явное взаимодействие между представлениями и контроллером на основе модели представления?
Ответ 3
@Майкл Уиллмотт
Хотя эта тема 6 лет, но я считаю ее полезной для сценария, с которым я недавно имел дело.
В моем представлении MVC вместо использования полного пути для идентификаторов я использовал встроенную короткую переменную с Razor, и это привело к тому, что модель обратного просмотра на сервере была нулевой. Это объясняется тем, что атрибуты "name" и "id" для элементов html не отображались правильно, поскольку механизм привязки MVC ожидает:
Ниже указано неверное объявление:
@{
var b = Model.Blog; // short cut caused problem of postback model binding
}
<div class="editor-field">
@Html.EditorFor(model => b.Title)
@Html.ValidationMessageFor(model => b.Title)
</div>
Ниже приведено правильное объявление, поскольку необходим полный путь для идентификаторов:
<div class="editor-field">
@Html.EditorFor(model => model.Blog.Title)
@Html.ValidationMessageFor(model => model.Blog.Title)
</div>