我正在处理已部分转换为MVC的现有应用程序。每当控制器以JSON ActionResult响应时,枚举都将以与字符串名称相对的数字发送。听起来默认的序列化程序应该是JSON.Net,应该将枚举作为其名称发送给整数表示形式,但事实并非如此。

我缺少web.config吗?设置,将其设置为默认序列化程序?还是还有其他需要更改的设置?

#1 楼

在ASP.Net MVC4中,在JsonResult类中使用的默认JavaScript序列化器仍然是JavaScriptSerializer(您可以在代码中检查它)。

我想您已经将它与ASP.Net Web混淆了。 API,其中JSON.Net是默认的JS序列化器,但MVC4不使用它。关于它的大量文章:


ASP.NET MVC和Json.NET
在ASP.NET MVC 3中使用JSON.NET作为默认JSON序列化程序-是否可能如果您还想将JSON.Net用于控制器操作参数,那么在模型绑定期间,则需要编写自己的JsonNetResult实现。

并且您需要在以下位置注册您的实现:

ValueProviderFactories.Factories
    .Remove(ValueProviderFactories.Factories
                                  .OfType<JsonValueProviderFactory>().Single());
ValueProviderFactories.Factories.Add(new MyJsonValueProviderFactory());


您可以使用内置的ValueProviderFactory作为示例或本文:ASP。 NET MVC 3 –使用Json.Net改进了JsonValueProviderFactory

评论


这有助于发送JSON响应,但不能反序列化传入的JSON,因为该框架尝试将传入的调用映射到传递给您的action方法的参数中。 。 。您如何更改MVC为此使用的串行器?

–blaster
13年10月6日,0:14

如果要将Json.NET用于传入参数,则需要编写自己的ValueProviderFactory实现。您可以使用内置的JsonValueProviderFactory作为示例。并且您需要向以下位置注册您的实现:ValueProviderFactories.Factories.Remove(ValueProviderFactories.Factories.OfType ()。Single()); ValueProviderFactories.Factories.Add(new MyJsonValueProviderFactory());。另请参阅:dalsoft.co.uk/blog/index.php/2012/01/10/…

–nemesv
13年10月6日在6:49

@Marc根据源代码:aspnetwebstack.codeplex.com/SourceControl/latest#src/…MVC5中的JsonResult仍然使用旧的JavaScriptSerializer,并且不使用JSON.net

–nemesv
2015年9月9日在18:59



@jpgrassi OfType是System.Linq命名空间中定义的扩展方法。你有使用System.Linq;在你的cs文件?

–nemesv
2015年9月24日在17:04

作为记录,MVC 6最终使用Json.NET作为默认记录。

–约翰
16年4月28日在14:30

#2 楼

ASP.NET MVC 5修复:

我还没有准备好更改为Json.NET,就我而言,该错误是在请求期间发生的。在我的方案中,最好的方法是修改实际的JsonValueProviderFactory,将其应用到全局项目,并可以通过编辑global.cs文件来完成。

JsonValueProviderConfig.Config(ValueProviderFactories.Factories);


添加网站.config条目:

<add key="aspnet:MaxJsonLength" value="20971520" />


,然后创建以下两个类

public class JsonValueProviderConfig
{
    public static void Config(ValueProviderFactoryCollection factories)
    {
        var jsonProviderFactory = factories.OfType<JsonValueProviderFactory>().Single();
        factories.Remove(jsonProviderFactory);
        factories.Add(new CustomJsonValueProviderFactory());
    }
}


这基本上是一个精确的过程在System.Web.Mvc中找到的默认实现的副本,但添加了可配置的web.config应用设置值aspnet:MaxJsonLength

public class CustomJsonValueProviderFactory : ValueProviderFactory
{

    /// <summary>Returns a JSON value-provider object for the specified controller context.</summary>
    /// <returns>A JSON value-provider object for the specified controller context.</returns>
    /// <param name="controllerContext">The controller context.</param>
    public override IValueProvider GetValueProvider(ControllerContext controllerContext)
    {
        if (controllerContext == null)
            throw new ArgumentNullException("controllerContext");

        object deserializedObject = CustomJsonValueProviderFactory.GetDeserializedObject(controllerContext);
        if (deserializedObject == null)
            return null;

        Dictionary<string, object> strs = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
        CustomJsonValueProviderFactory.AddToBackingStore(new CustomJsonValueProviderFactory.EntryLimitedDictionary(strs), string.Empty, deserializedObject);

        return new DictionaryValueProvider<object>(strs, CultureInfo.CurrentCulture);
    }

    private static object GetDeserializedObject(ControllerContext controllerContext)
    {
        if (!controllerContext.HttpContext.Request.ContentType.StartsWith("application/json", StringComparison.OrdinalIgnoreCase))
            return null;

        string fullStreamString = (new StreamReader(controllerContext.HttpContext.Request.InputStream)).ReadToEnd();
        if (string.IsNullOrEmpty(fullStreamString))
            return null;

        var serializer = new JavaScriptSerializer()
        {
            MaxJsonLength = CustomJsonValueProviderFactory.GetMaxJsonLength()
        };
        return serializer.DeserializeObject(fullStreamString);
    }

    private static void AddToBackingStore(EntryLimitedDictionary backingStore, string prefix, object value)
    {
        IDictionary<string, object> strs = value as IDictionary<string, object>;
        if (strs != null)
        {
            foreach (KeyValuePair<string, object> keyValuePair in strs)
                CustomJsonValueProviderFactory.AddToBackingStore(backingStore, CustomJsonValueProviderFactory.MakePropertyKey(prefix, keyValuePair.Key), keyValuePair.Value);

            return;
        }

        IList lists = value as IList;
        if (lists == null)
        {
            backingStore.Add(prefix, value);
            return;
        }

        for (int i = 0; i < lists.Count; i++)
        {
            CustomJsonValueProviderFactory.AddToBackingStore(backingStore, CustomJsonValueProviderFactory.MakeArrayKey(prefix, i), lists[i]);
        }
    }

    private class EntryLimitedDictionary
    {
        private static int _maximumDepth;

        private readonly IDictionary<string, object> _innerDictionary;

        private int _itemCount;

        static EntryLimitedDictionary()
        {
            _maximumDepth = CustomJsonValueProviderFactory.GetMaximumDepth();
        }

        public EntryLimitedDictionary(IDictionary<string, object> innerDictionary)
        {
            this._innerDictionary = innerDictionary;
        }

        public void Add(string key, object value)
        {
            int num = this._itemCount + 1;
            this._itemCount = num;
            if (num > _maximumDepth)
            {
                throw new InvalidOperationException("The length of the string exceeds the value set on the maxJsonLength property.");
            }
            this._innerDictionary.Add(key, value);
        }
    }

    private static string MakeArrayKey(string prefix, int index)
    {
        return string.Concat(prefix, "[", index.ToString(CultureInfo.InvariantCulture), "]");
    }

    private static string MakePropertyKey(string prefix, string propertyName)
    {
        if (string.IsNullOrEmpty(prefix))
        {
            return propertyName;
        }
        return string.Concat(prefix, ".", propertyName);
    }

    private static int GetMaximumDepth()
    {
        int num;
        NameValueCollection appSettings = ConfigurationManager.AppSettings;
        if (appSettings != null)
        {
            string[] values = appSettings.GetValues("aspnet:MaxJsonDeserializerMembers");
            if (values != null && values.Length != 0 && int.TryParse(values[0], out num))
            {
                return num;
            }
        }
        return 1000;
    }

    private static int GetMaxJsonLength()
    {
        int num;
        NameValueCollection appSettings = ConfigurationManager.AppSettings;
        if (appSettings != null)
        {
            string[] values = appSettings.GetValues("aspnet:MaxJsonLength");
            if (values != null && values.Length != 0 && int.TryParse(values[0], out num))
            {
                return num;
            }
        }
        return 1000;
    }
}


评论


在我们的应用程序中添加了这两个类,它可以正常工作。非常感谢,伙计,您为我们节省了很多时间和麻烦。

–Lodoss
19年3月13日在15:14

if(num> _maximumDepth)中的错误消息中有错误。它不是MaxJsonLength,而是MaxJsonDeserializerMembers属性。您的代码救了我。 ;)

– Evilripper
19年7月19日在8:51