ASP.Net MVC默認的JSON序列化使用的是微軟自己的JavaScriptSerializer。性能低不說,最讓人受不了的是Dictionary<,>和Hashtable類型居然對應的json是[{"Key":"a","Value":1}]而不是{"a":1}。真是奇葩到家了,跟前端根本沒法集成!
決定還是用JSON.Net來做吧!查了各種資料,網上的要么代碼太復雜,要么沒法完全實現。又在ILSpy分析了MVC源碼,然后調試了半天,終於有了初步的解決辦法:
1、反序列化,首先建立下面的類:
原理基本上就是替換系統自帶的ValueProviderFactory和DefaultModelBinder,然后改用自己的類來調用JSON.Net實現。
public class JsonNetValueProviderFactory : ValueProviderFactory { public override IValueProvider GetValueProvider(ControllerContext ctlContext) { //if (!controllerContext.HttpContext.Request.ContentType. // StartsWith("application/json", StringComparison.OrdinalIgnoreCase)) //{ // return null; //} var reader = new StreamReader(ctlContext.HttpContext.Request.InputStream); reader.BaseStream.Position = 0; var json = reader.ReadToEnd()?.TrimStart(' ', '\r', '\n', '\t'); if (string.IsNullOrEmpty(json)) return null; var jtoken = json.StartsWith("[") ? JArray.Parse(json) as JContainer : JObject.Parse(json) as JContainer; return new JsonNetValueProvider(jtoken); } } public class JsonNetValueProvider : IValueProvider { private JContainer _jValue; public JsonNetValueProvider(JContainer jval) { _jValue = jval; } public bool ContainsPrefix(string prefix) { return true; } public ValueProviderResult GetValue(string key) { var jtoken = _jValue.SelectToken(key); if (jtoken == null) { jtoken = _jValue; } return new JsonNetValueProviderResult(jtoken, key, null); } } public class JsonNetValueProviderResult : ValueProviderResult { private JToken _jtoken; public JsonNetValueProviderResult(JToken valueRaw, string key, CultureInfo info) { _jtoken = valueRaw; } [System.Diagnostics.DebuggerHidden] public override object ConvertTo(Type type, CultureInfo culture) { return _jtoken?.ToObject(type); } } public class JsonNetModelBinder : DefaultModelBinder { [System.Diagnostics.DebuggerHidden] public override object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext) { var provider = bindingContext.ValueProvider.GetValue(bindingContext.ModelName); if (provider != null) { try { return provider.ConvertTo(bindingContext.ModelType); } catch { } } return base.BindModel(controllerContext, bindingContext); } }
2、然后再Application_Start或者Router初始化時調用下面代碼:
//重置Json序列化方式,改用JSON.Net ValueProviderFactories.Factories.Remove(ValueProviderFactories.Factories. OfType<JsonValueProviderFactory>().FirstOrDefault()); ValueProviderFactories.Factories.Add(new JsonNetValueProviderFactory()); //重置系統的Binder,使Dictionary可以正常json ModelBinders.Binders.DefaultBinder = new JsonNetModelBinder();
3、序列化的過程比較簡單,建立一個JsonNetResult類,然后mvc方法返回這個類型就可以了。
public class JsonNetResult : JsonResult { public JsonSerializerSettings Settings { get; private set; } public override void ExecuteResult(ControllerContext context) { //if (context == null) // throw new ArgumentNullException("context"); //if (this.JsonRequestBehavior == JsonRequestBehavior.DenyGet && string.Equals(context.HttpContext.Request.HttpMethod, "GET", StringComparison.OrdinalIgnoreCase)) // throw new InvalidOperationException("JSON GET is not allowed"); HttpResponseBase response = context.HttpContext.Response; response.ContentType = string.IsNullOrEmpty(this.ContentType) ? "application/json" : this.ContentType; if (this.ContentEncoding != null) response.ContentEncoding = this.ContentEncoding; if (this.Data == null) return; var scriptSerializer = JsonSerializer.Create(this.Settings); using (var sw = new StringWriter()) { scriptSerializer.Serialize(sw, this.Data); response.Write(sw.ToString()); } } }
