使用JSON.Net(Newtonsoft.Json)作為ASP.Net MVC的json序列化和反序列化工具


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());
            }
        }
    }

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM