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>