我缺少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 #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
评论
这有助于发送JSON响应,但不能反序列化传入的JSON,因为该框架尝试将传入的调用映射到传递给您的action方法的参数中。 。 。您如何更改MVC为此使用的串行器?
–blaster
13年10月6日,0:14
如果要将Json.NET用于传入参数,则需要编写自己的ValueProviderFactory实现。您可以使用内置的JsonValueProviderFactory作为示例。并且您需要向以下位置注册您的实现:ValueProviderFactories.Factories.Remove(ValueProviderFactories.Factories.OfType
–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