2010-03-11 124 views
3

我想使用jQuery發佈一些具有ID和某些類別的產品。但是,當我添加Categories時,出現以下錯誤:Microsoft.Web.Mvc.DataAnnotations.DataAnnotationsModelBinder.BindProperty(NullReferenceException)。如何使用jQuery發佈複雜對象(包含複雜對象數組)的數組

不應該默認的ModelBinder能夠綁定這個(沒有ActionFilter或自定義ModelBinder)?

我試圖應用一個ActionFilter(反序列化),我發現在另一個SO線程,但它永遠不會運行。我也試過用jQuery.ajaxSettings.traditional = true,jQuery 1.3.2和1.4.2。在我發現的其他例子中,他們只是發佈ID,Name等,而不是另一個複雜對象的數組。

任何想法?

public class Product 
{ 
    public int ID { get; set; } 
    public Category[] Categories { get; set; } 
} 

public class Category 
{ 
    public int ID { get; set; } 
} 

HTML

<input id="Product[0]_ID" name="Product[0].ID" type="hidden" value="9" /> 
<input id="Product[0]_Categories[0]_ID" name="Product[0].Categories[0].ID" type="hidden" value="99" /> 
<input id="Product[1]_ID" name="Product[1].ID" type="hidden" value="8" /> 
<input id="Product[1]_Categories[0]_ID" name="Product[1].Categories[0].ID" type="hidden" value="88" /> 

控制器

[JsonFilter(Parameter = "p")] 
public JsonResult GetProductPrice([Bind(Prefix = "Product")] Product[] p) 
{ 
    // TODO: Implement some checking... 
    return Json(true); 
} 

jQuery的

$.post(getProductPriceUrl, $("form").serializeArray(), function(data) { 
    $("#Price").html(data); 
}); 

JsonFilter

public class JsonFilter : ActionFilterAttribute 
{ 
    public string Parameter { get; set; } 
    //public Type JsonDataType { get; set; } 

    private JavaScriptSerializer serializer; 

    public override void OnActionExecuting(ActionExecutingContext filterContext) 
    { 
     serializer = new JavaScriptSerializer(); 

     if (filterContext.HttpContext.Request.ContentType.Equals("application/json")) 
     { 
      string inputContent; 

      using (var sr = new StreamReader(filterContext.HttpContext.Request.InputStream)) 
      { 
       inputContent = sr.ReadToEnd(); 
      } 

      var result = serializer.Deserialize<Product>(inputContent); 
      filterContext.ActionParameters[Parameter] = result; 
     } 
    } 
} 

POST#1

__RequestVerificationToken=sz%2BLKCzTmdGMrH3TdOYipS5z%2BJ3uVyzBtJRZrruJoUohoGaH2O3DU5%2FcuU6hX1E%2F&Product%5B0%5D.ID=9&Product%5B0%5D.Categories%5B0%5D.ID=99&Product%5B1%5D.ID=8&Product%5B1%5D.Categories%5B0%5D.ID=88 

POST#2

__RequestVerificationToken=sz+LKCzTmdGMrH3TdOYipWTERHSdtCvGUhuw/dGIkgSL3rjcSLO7RJJN/rcssVwv&Product[0].ID=9&Product[0].Categories[0].ID=99&Product[1].ID=8&Product[1].Categories[0].ID=88 

POST#3

[{"name":"__RequestVerificationToken","value":"sz+LKCzTmdGMrH3TdOYipcqr8WKC2eL7CRS5BZUtwzD60WkqfnjdeAcO3DQg5ss6"},{"name":"Product[0].ID","value":"9"},{"name":"Product[0].Categories[0].ID","value":"99"},{"name":"Product[1].ID","value":"8"},{"name":"Product[1].Categories[0].ID","value":"88"}] 
+0

你不應該需要一個ActionFilter。你可以顯示「POST」數據嗎? – 2010-03-11 13:53:13

+0

呃,帖子1&2是常規鍵/值,「帖子」3是JSON。你打算做什麼?爲什麼有兩種方法?一般來說,首選MVC中的標準POST,因爲JSON需要特殊處理。 – 2010-03-11 16:58:42

+0

我想使用一個標準的POST,但經過幾個小時的測試,並沒有得到它的工作。我開始懷疑這是可能的,所以我用這個解決方案閱讀了一些SO線程後,嘗試了JSON和「JsonFilter」。 – 2010-03-11 20:05:55

回答

0

有兩種方法我能想到的來完成發送一個複雜的對象圖到目標的目標服務器連接正確。一種是你最初嘗試的方式,那就是在html中表達關係。另一種是傳遞給子對象的部分視圖。

因此,您需要爲類別列表創建部分視圖並將Product.categories傳遞給它。這應該照顧好你的線路。

這將是

@Html.Partial("_myCategoriesPartialView", Model.Categories) 

當然,這種觀點會遍歷集合。 我相信如果你想用html的方式做它,它會看起來像。

<input id="Product[0]_ID" name="Product[0].ID" type="hidden" value="9" /> 

應該

<input id="Product[0].ID" name="Product[0].ID" type="hidden" value="9" />, 

但肯定地告訴的方法是看Response.Form數據