1

我正在尝试设置一个编辑视图,在该视图上我有一个文本框和 DropDownListFor。我已经找到了一种填充 DDLF 的方法,并且渲染和发布的值是正确的,但我似乎无法让模型正确更新。

我试图更新的对象是从 LINQtoSQL 生成的,并且在数据库中它具有外键列。在导致“包含”关系的 LINQtoSQL 类中。我可以访问代表 DB 中的列的 ID 属性,以及它代表的对象。

        zupanija = new Zupanija();       //object that needs to be updated
        zupanija.Drzava;                 //object that i want to change to make the update
        zupanija.DrzavaID;               //Property linked to object that should change

我想出进行更新的唯一方法是从 DDLF 获取值并使用它来获取我想要更改的对象,如下所示:

        [HttpPost]
        public ActionResult Edit(int id, FormCollection collection)
        {
         var zupanija = repo.ZupanijaById(id);
         var drzava = new repoDrzava().DrzavaById(Convert.ToInt32(collection["Zupanija.DrzavaID"]));
         zupanija.Drzava = drzava;
        }

此外,当我尝试像这样更新 ID 字段时,我得到以下错误:

           zupanija.DrzavaID = Convert.ToInt32(collection["Zupanija.DrzavaID"]);

错误:抛出新的 System.Data.Linq.ForeignKeyReferenceAlreadyHasValueException();

在我看来,这样做是非常糟糕的方式,我正试图让 UpdateModel 工作。

4

2 回答 2

5

我在寻找其他东西的同时找到了解决方案,在 Joe Stevens 的博客中:

使用 ViewModel 时使用 Controller UpdateModel

问题如下:当使用视图模型正确绑定属性时,有必要“指导”UpdateModel 助手如何找到我们希望更新的实际类。

我的解决方案需要修改

 UpdateModel(zupanija); to  UpdateModel(zupanija,"Zupanija");

因为我使用的 ViewModel 类包含几个属性以及我想要更新的主要数据类。下面是代码,希望对理解有帮助:

    public class ZupanijaFVM
    {
    public IEnumerable<SelectListItem> Drzave { get; private set; }
    public Zupanija Zupanija { get; private set; }
    ...
    }

    // From Controller
    //
    // GET: /Admin/Zupanije/Edit/5
    public ActionResult Edit(int id)
    {
        var zupanija = repo.ZupanijaById(id);
        return zupanija == null ? View("Error") : View(new ZupanijaFVM(repo.ZupanijaById(id)));
    }

    //
    // POST: /Admin/Zupanije/Edit/5

    [HttpPost]
    public ActionResult Edit(int id, FormCollection collection)
    {
        var zupanija = repo.ZupanijaById(id);
        if (TryUpdateModel(zupanija, "Zupanija"))
        {
            repo.Save();
            return RedirectToAction("Details", new { id = zupanija.ZupanijaID });
        }
        return View(new ZupanijaFVM(zupanija));
    }

     //From View:
     @model VozniRed.Areas.Admin.Models.ZupanijeFVM

     <script src="@Url.Content("~/Scripts/jquery.validate.min.js")" type="text/javascript"></script>
     <script src="@Url.Content("~/Scripts/jquery.validate.unobtrusive.min.js")" type="text/javascript"></script>
      @using (Html.BeginForm()) 
      {
        @Html.ValidationSummary(true)
        <fieldset>
         <legend>Zupanija</legend>
          @Html.HiddenFor(model => model.Zupanija.ZupanijaID)
         <div class="editor-label">
          @Html.LabelFor(model => model.Zupanija.Naziv)
         </div>
         <div class="editor-field">
          @Html.EditorFor(model => model.Zupanija.Naziv)
          @Html.ValidationMessageFor(model => model.Zupanija.Naziv)
         </div>
         <div class="editor-label">
          @Html.LabelFor(model => model.Zupanija.Drzava)
         </div>
         <div class="editor-field">
          @Html.DropDownListFor(model => model.Zupanija.DrzavaID, Model.Drzave)
          @Html.ValidationMessageFor(model => model.Zupanija.DrzavaID)
         </div>
         <p>
          <input type="submit" value="Save" />
         </p>
       </fieldset>
       }
    <div>
     @Html.ActionLink("Back to List", "Index")
    </div>
于 2011-05-01T22:36:43.047 回答
1

下拉列表由<select>HTML 表单中的标记表示。A<select>包含一个标签列表,<option>每个标签都包含一个 ID 和一个文本。当用户选择一个选项并提交表单时,该选项的相应 ID 将被 POST 到服务器。而且只有身份证。因此,您可以期望在EditPOST 操作中获得所选选项的 ID。所做UpdateModel的只是使用发送的请求参数并将它们转换为强类型对象。但是因为所有 POSTed 都是一个简单的 ID,这就是你所能得到的。从那里开始,如果您想获取相应的模型,则必须使用此 ID 查询数据存储。所以你不能得到不存在的东西。

于 2011-05-01T19:19:34.853 回答