アップデート
Jeroen によるブログ投稿 (以下のリンク付きの彼の回答を参照) と、自分のコードを再確認した後の頭のひらめきに基づいて、ExtendedJsonValueProviderFactoryを更新し、トップレベルの BackingStore が常に適切に作成されるようにしました。 JSON 経由で送信された辞書。
このコードは GitHub のhttps://github.com/counsellorben/ASP.NET-MVC-JsonDictionaryBindingで入手でき、実際の例はhttp://oss.form.vu/json-dictionary-example/にあります。
現在のものを削除して、JsonValueProviderFactory
辞書の作成を処理できるものに置き換えることで、辞書をバインドできます。まず、Keith が指摘したように、Javascript では、必ず "filterItem" 内に辞書をラップしてください。これは、コントローラー アクションのモデル変数の名前であり、JSON の場合、コントローラー アクションの変数の名前だからです。返される Json 要素の名前と一致する必要があります。また、クラスを渡す場合、ネストされた要素はクラス内のプロパティの名前と一致する必要があります。
次に、次のようにExtendedJsonValueProviderFactory
クラスを作成します。
using System;
using System.Collections;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Globalization;
using System.IO;
using System.Web.Script.Serialization;
public sealed class ExtendedJsonValueProviderFactory : ValueProviderFactory
{
private void AddToBackingStore(Dictionary<string, object> backingStore, string prefix, object value)
{
IDictionary<string, object> d = value as IDictionary<string, object>;
if (d != null)
{
foreach (KeyValuePair<string, object> entry in d)
{
if (entry.Key.EndsWith("Dictionary", StringComparison.CurrentCulture))
CreateDictionary(backingStore, entry);
else
AddToBackingStore(backingStore, MakePropertyKey(prefix, entry.Key), entry.Value);
}
return;
}
IList l = value as IList;
if (l != null)
{
for (int i = 0; i < l.Count; i++)
{
AddToBackingStore(backingStore, MakeArrayKey(prefix, i), l[i]);
}
return;
}
// primitive
backingStore[prefix] = value;
}
private void CreateDictionary(Dictionary<string, object> backingStore, KeyValuePair<string, object> source)
{
var d = source.Value as IDictionary<string, object>;
var dictionary = new Dictionary<string, string>();
foreach (KeyValuePair<string, object> entry in d)
dictionary.Add(entry.Key, entry.Value.ToString());
AddToBackingStore(backingStore, source.Key, dictionary);
return;
}
private static object GetDeserializedObject(ControllerContext controllerContext)
{
if (!controllerContext.HttpContext.Request.ContentType.StartsWith("application/json", StringComparison.OrdinalIgnoreCase))
{
// not JSON request
return null;
}
StreamReader reader = new StreamReader(controllerContext.HttpContext.Request.InputStream);
string bodyText = reader.ReadToEnd();
if (String.IsNullOrEmpty(bodyText))
{
// no JSON data
return null;
}
JavaScriptSerializer serializer = new JavaScriptSerializer();
object jsonData = serializer.DeserializeObject(bodyText);
return jsonData;
}
public override IValueProvider GetValueProvider(ControllerContext controllerContext)
{
if (controllerContext == null)
{
throw new ArgumentNullException("controllerContext");
}
object jsonData = GetDeserializedObject(controllerContext);
if (jsonData == null)
{
return null;
}
Dictionary<string, object> backingStore = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
AddToBackingStore(backingStore, String.Empty, jsonData);
return new DictionaryValueProvider<object>(backingStore, CultureInfo.CurrentCulture);
}
private static string MakeArrayKey(string prefix, int index)
{
return prefix + "[" + index.ToString(CultureInfo.InvariantCulture) + "]";
}
private static string MakePropertyKey(string prefix, string propertyName)
{
return (String.IsNullOrEmpty(prefix)) ? propertyName : prefix + "." + propertyName;
}
}
このクラスは、 type の DictionaryValueProvider へのエントリを構築するための拡張機能を除いて、標準の JsonValueProviderFactory クラスとほとんど同じであることに気付くかもしれませんDictionary<string,string>
。また、ディクショナリとして処理されるためには、要素の名前が「Dictionary」で終わる必要があることにも注意してください(これは重大なコード臭だと思いますが、現時点では別の代替案を思いつきません ...私は提案を受け入れています)。
次に、以下をApplication_Start
inに追加しGlobal.asax.cs
ます。
var j = ValueProviderFactories.Factories.FirstOrDefault(f => f.GetType().Equals(typeof(JsonValueProviderFactory)));
if (j != null)
ValueProviderFactories.Factories.Remove(j);
ValueProviderFactories.Factories.Add(new ExtendedJsonValueProviderFactory());
これにより、標準の JsonValueProviderFactory が削除され、拡張クラスに置き換えられます。
最後のステップ:良さを楽しむ。