首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >如何在MVC5项目中使用Json.NET进行JSON模型绑定?

如何在MVC5项目中使用Json.NET进行JSON模型绑定?
EN

Stack Overflow用户
提问于 2014-06-02 21:02:29
回答 2查看 24.6K关注 0票数 28

我一直在互联网上寻找答案或例子,但还没有找到。我只想更改默认的JSON序列化器,它用于在模型绑定到JSON.NET库时反序列化JSON。

我已经找到了this所以post,但到目前为止还不能实现它,我甚至看不到System.Net.Http.Formatters名称空间,也看不到GlobalConfiguration

我遗漏了什么?

更新

我有一个ASP.NET MVC项目,它基本上是一个MVC3项目。目前我的目标是.NET 4.5,并使用ASP.NET MVC5和相关的NuGet包。

我没有看到System.Web.Http程序集,也没有看到任何类似的命名空间。在这个上下文中,我想注入JSON.NET作为JSON类型请求的默认模型绑定器。

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2014-06-03 17:50:01

我终于找到了答案。基本上我不需要MediaTypeFormatter的东西,它不是为在MVC环境中使用而设计的,但在ASP.NET Web中,这就是为什么我看不到那些引用和名称空间(顺便说一句,这些都包含在Microsoft.AspNet.WeApi NuGet包中)。

解决方案是使用自定义值提供程序工厂。以下是所需的代码。

代码语言:javascript
复制
    public class JsonNetValueProviderFactory : ValueProviderFactory
    {
        public override IValueProvider GetValueProvider(ControllerContext controllerContext)
        {
            // first make sure we have a valid context
            if (controllerContext == null)
                throw new ArgumentNullException("controllerContext");

            // now make sure we are dealing with a json request
            if (!controllerContext.HttpContext.Request.ContentType.StartsWith("application/json", StringComparison.OrdinalIgnoreCase))
                return null;

            // get a generic stream reader (get reader for the http stream)
            var streamReader = new StreamReader(controllerContext.HttpContext.Request.InputStream);
            // convert stream reader to a JSON Text Reader
            var JSONReader = new JsonTextReader(streamReader);
            // tell JSON to read
            if (!JSONReader.Read())
                return null;

            // make a new Json serializer
            var JSONSerializer = new JsonSerializer();
            // add the dyamic object converter to our serializer
            JSONSerializer.Converters.Add(new ExpandoObjectConverter());

            // use JSON.NET to deserialize object to a dynamic (expando) object
            Object JSONObject;
            // if we start with a "[", treat this as an array
            if (JSONReader.TokenType == JsonToken.StartArray)
                JSONObject = JSONSerializer.Deserialize<List<ExpandoObject>>(JSONReader);
            else
                JSONObject = JSONSerializer.Deserialize<ExpandoObject>(JSONReader);

            // create a backing store to hold all properties for this deserialization
            var backingStore = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
            // add all properties to this backing store
            AddToBackingStore(backingStore, String.Empty, JSONObject);
            // return the object in a dictionary value provider so the MVC understands it
            return new DictionaryValueProvider<object>(backingStore, CultureInfo.CurrentCulture);
        }

        private static void AddToBackingStore(Dictionary<string, object> backingStore, string prefix, object value)
        {
            var d = value as IDictionary<string, object>;
            if (d != null)
            {
                foreach (var entry in d)
                {
                    AddToBackingStore(backingStore, MakePropertyKey(prefix, entry.Key), entry.Value);
                }
                return;
            }

            var l = value as IList;
            if (l != null)
            {
                for (var i = 0; i < l.Count; i++)
                {
                    AddToBackingStore(backingStore, MakeArrayKey(prefix, i), l[i]);
                }
                return;
            }

            // primitive
            backingStore[prefix] = value;
        }

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

您可以在Application_Start方法中像这样使用它:

代码语言:javascript
复制
// remove default implementation    
ValueProviderFactories.Factories.Remove(ValueProviderFactories.Factories.OfType<JsonValueProviderFactory>().FirstOrDefault());
// add our custom one
ValueProviderFactories.Factories.Add(new JsonNetValueProviderFactory());

Here为我指明了正确的方向,also this one对值提供者和模型绑定器做了很好的解释。

票数 30
EN

Stack Overflow用户

发布于 2015-11-02 20:51:42

在我的例子中,我必须反序列化复杂的对象,包括接口和动态加载的类型等,所以提供自定义值提供程序不起作用,因为MVC仍然需要找出如何实例化接口,然后失败。

由于我的对象已经正确地添加了注释以使用Json.NET,因此我采取了一种不同的方法:我实现了一个自定义模型绑定器,并使用Json.NET显式地反序列化请求体数据,如下所示:

代码语言:javascript
复制
internal class CustomModelBinder : IModelBinder
{
    public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        // use Json.NET to deserialize the incoming Position
        controllerContext.HttpContext.Request.InputStream.Position = 0; // see: http://stackoverflow.com/a/3468653/331281
        Stream stream = controllerContext.RequestContext.HttpContext.Request.InputStream;
        var readStream = new StreamReader(stream, Encoding.UTF8);
        string json = readStream.ReadToEnd();
        return JsonConvert.DeserializeObject<MyClass>(json, ...);
    }
}

自定义模型绑定器在Global.asax.cs中注册

代码语言:javascript
复制
  ModelBinders.Binders.Add(typeof(MyClass), new CustomModelBinder();
票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/23995210

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档