JavaScriptSerializer期间ASP.NET MVC中的MaxJsonLength异常


122

在我的控制器动作之一中,我将返回一个非常大的值JsonResult以填充网格。

我收到以下InvalidOperationException异常:

使用JSON JavaScriptSerializer进行序列化或反序列化时出错。字符串的长度超过了在maxJsonLength属性上设置的值。

不幸的是,将中的maxJsonLength属性设置web.config为较高的值不会显示任何效果。

<system.web.extensions>
  <scripting>
    <webServices>
      <jsonSerialization maxJsonLength="2147483644"/>
    </webServices>
  </scripting>
</system.web.extensions>

我不想像这个 SO答案中提到的那样将其作为字符串传递回来。

在我的研究中,我偶然发现了博客文章,建议您编写自己的文章ActionResult(例如LargeJsonResult : JsonResult)以绕过此行为。

这是唯一的解决方案吗?
这是ASP.NET MVC中的错误吗?
我想念什么吗?

非常感激任何的帮助。


2
您的解决方案适用于MVC 3
MatteoSp

1
@Matteo确定吗?这是一个老问题,我不记得了,但显然我将其标记为MVC3。不幸的是,当它固定/关闭时,我看不到版本/日期:aspnet.codeplex.com/workitem/3436
Martin Buberl 2013年

1
当然,我正在使用MVC 3,并且可以使用。幸运的是,由于在MVC 3中没有公认的答案中引用的“ MaxJsonLength”属性。
MatteoSp

Answers:


228

看来这已在MVC4中修复。

您可以这样做,对我来说效果很好:

public ActionResult SomeControllerAction()
{
  var jsonResult = Json(veryLargeCollection, JsonRequestBehavior.AllowGet);
  jsonResult.MaxJsonLength = int.MaxValue;
  return jsonResult;
}

我正在将json字符串设置为ViewBag.MyJsonString属性,但在运行时在以下JavaScript行上的视图中出现了相同的错误:var myJsonObj = @ Html.Raw(Json.Encode(ViewBag.MyJsonString));
Faisal Mq 2013年

1
嘿@orionedvards,@ GG,@ MartinBuberl我正面临着同样的maxJson问题,但是当将数据发布到控制器时,我该如何处理这个问题,我花了很多时间进行搜索。任何帮助将不胜感激。
Katmanco'3

在我的情况下没有用,因为我必须在json序列化集合之前设置MaxJsonLength。
塞萨尔·莱昂

就我的情况而言,它工作得很好,因为数据表中的“ IMAGES”要呈现给最终用户,所以我不得不实施它。没有它,只是崩溃而没有任何可理解的消息。
毛罗·坎迪多

33

您也可以ContentResult按照此处的建议使用,而不是子类化JsonResult

var serializer = new JavaScriptSerializer { MaxJsonLength = Int32.MaxValue, RecursionLimit = 100 };

return new ContentResult()
{
    Content = serializer.Serialize(data),
    ContentType = "application/json",
};

2
以我为例,在一个一次性应用程序上工作时,此解决方案对我来说效果最好。保存了实现的jsonresult。谢谢!
克里斯托(Christo)2012年


22

无需自定义类。这就是所需要的:

return new JsonResult { Data = Result, MaxJsonLength = Int32.MaxValue };

Result您要序列化的数据在哪里。


错误137'System.Web.Mvc.JsonResult'不包含'MaxJsonLength'的定义
PUG

这对我有用,但是仍然需要添加:JsonRequestBehavior = JsonRequestBehavior.AllowGet
DubMan

5

如果使用Json.NET生成json字符串,则不需要设置MaxJsonLength值。

return new ContentResult()
{
    Content = Newtonsoft.Json.JsonConvert.SerializeObject(data),
    ContentType = "application/json",
};

4

我通过以下 链接解决了这个问题

namespace System.Web.Mvc
{
public sealed class JsonDotNetValueProviderFactory : ValueProviderFactory
{
    public override IValueProvider GetValueProvider(ControllerContext controllerContext)
    {
        if (controllerContext == null)
            throw new ArgumentNullException("controllerContext");

        if (!controllerContext.HttpContext.Request.ContentType.StartsWith("application/json", StringComparison.OrdinalIgnoreCase))
            return null;

        var reader = new StreamReader(controllerContext.HttpContext.Request.InputStream);
        var bodyText = reader.ReadToEnd();

        return String.IsNullOrEmpty(bodyText) ? null : new DictionaryValueProvider<object>(JsonConvert.DeserializeObject<ExpandoObject>(bodyText, new ExpandoObjectConverter()), CultureInfo.CurrentCulture);
    }
}

}

protected void Application_Start()
    {
        AreaRegistration.RegisterAllAreas();

        RegisterGlobalFilters(GlobalFilters.Filters);
        RegisterRoutes(RouteTable.Routes);

        //Remove and JsonValueProviderFactory and add JsonDotNetValueProviderFactory
        ValueProviderFactories.Factories.Remove(ValueProviderFactories.Factories.OfType<JsonValueProviderFactory>().FirstOrDefault());
        ValueProviderFactories.Factories.Add(new JsonDotNetValueProviderFactory());
    }

3

我很惊讶没有人建议使用结果过滤器。这是全局连接到操作/结果管道的最干净的方法:

public class JsonResultFilter : IResultFilter
{
    public int? MaxJsonLength { get; set; }

    public int? RecursionLimit { get; set; }

    public void OnResultExecuting(ResultExecutingContext filterContext)
    {
        if (filterContext.Result is JsonResult jsonResult)
        {
            // override properties only if they're not set
            jsonResult.MaxJsonLength = jsonResult.MaxJsonLength ?? MaxJsonLength;
            jsonResult.RecursionLimit = jsonResult.RecursionLimit ?? RecursionLimit;
        }
    }

    public void OnResultExecuted(ResultExecutedContext filterContext)
    {
    }
}

然后,使用以下命令注册该类的实例GlobalFilters.Filters

GlobalFilters.Filters.Add(new JsonResultFilter { MaxJsonLength = int.MaxValue });

2

您可以尝试在LINQ表达式中仅定义所需的字段。

例。假设您有一个带有ID,名称,电话和图片的模型(字节数组)并且需要从json加载到选择列表中。

LINQ查询:

var listItems = (from u in Users where u.name.Contains(term) select u).ToList();

这里的问题是“ 选择你获取所有字段的 ”。因此,如果您有大图片,请帮我。

怎么解决?非常非常简单

var listItems = (from u in Users where u.name.Contains(term) select new {u.Id, u.Name}).ToList();

最佳做法是仅选择要使用的字段。

记得。这是一个简单的技巧,但可以帮助许多ASP.NET MVC开发人员。


1
在这种情况下,我不认为用户想要过滤其数据。某些人要求从数据库中带回大量行...
Simon Nicholls

2

替代ASP.NET MVC 5修复:

就我而言,该错误是在请求期间发生的。在我的方案中,最好的方法是修改将JsonValueProviderFactory修订应用于全局项目的实际方法,并且可以通过这样编辑global.cs文件来完成。

JsonValueProviderConfig.Config(ValueProviderFactories.Factories);

添加一个web.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 appsetting值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;
    }
}

非常感谢 !
larizzatg

1

在我将Action更改为之前,上述方法均无法解决[HttpPost]。并将ajax类型设置为POST

    [HttpPost]
    public JsonResult GetSelectedSignalData(string signal1,...)
    {
         JsonResult result = new JsonResult();
         var signalData = GetTheData();
         try
         {
              var serializer = new System.Web.Script.Serialization.JavaScriptSerializer { MaxJsonLength = Int32.MaxValue, RecursionLimit = 100 };

            result.Data = serializer.Serialize(signalData);
            return Json(result, JsonRequestBehavior.AllowGet);
            ..
            ..
            ...

    }

和ajax称为

$.ajax({
    type: "POST",
    url: some_url,
    data: JSON.stringify({  signal1: signal1,.. }),
    contentType: "application/json; charset=utf-8",
    success: function (data) {
        if (data !== null) {
            setValue();
        }

    },
    failure: function (data) {
        $('#errMessage').text("Error...");
    },
    error: function (data) {
        $('#errMessage').text("Error...");
    }
});

1
    protected override JsonResult Json(object data, string contentType, System.Text.Encoding contentEncoding, JsonRequestBehavior behavior)
    {
        return new JsonResult()
        {
            Data = data,
            ContentType = contentType,
            ContentEncoding = contentEncoding,
            JsonRequestBehavior = behavior,
            MaxJsonLength = Int32.MaxValue
        };
    }

是我在MVC 4中修复的问题吗?


0

您需要在配置代码返回JsonResult对象之前手动阅读配置部分。只需单行从web.config中读取:

        var jsonResult = Json(resultsForAjaxUI);
        jsonResult.MaxJsonLength = (ConfigurationManager.GetSection("system.web.extensions/scripting/webServices/jsonSerialization") as System.Web.Configuration.ScriptingJsonSerializationSection).MaxJsonLength;
        return jsonResult;

确保在web.config中定义了配置元素


0

这对我有用

        JsonSerializerSettings json = new JsonSerializerSettings
        {
            ReferenceLoopHandling = ReferenceLoopHandling.Ignore
        };
        var result = JsonConvert.SerializeObject(list, Formatting.Indented, json);
        return new JsonResult { Data = result, MaxJsonLength = int.MaxValue };

0

还有其他情况-数据从客户端发送到服务器。当您使用控制器方法并且模型很大时:

    [HttpPost]
    public ActionResult AddOrUpdateConsumerFile(FileMetaDataModelView inputModel)
    {
        if (inputModel == null) return null;
     ....
    }

系统抛出诸如“使用JSON JavaScriptSerializer进行序列化或反序列化时出错。字符串的长度超过在maxJsonLength属性上设置的值。参数名称:input”

在这种情况下,仅更改Web.config设置不足以提供帮助。您还可以重写mvc json序列化程序,以支持巨大的数据模型大小,或从Request中手动反序列化模型。您的控制器方法变为:

   [HttpPost]
    public ActionResult AddOrUpdateConsumerFile()
    {
        FileMetaDataModelView inputModel = RequestManager.GetModelFromJsonRequest<FileMetaDataModelView>(HttpContext.Request);
        if (inputModel == null) return null;
        ......
    }

   public static T GetModelFromJsonRequest<T>(HttpRequestBase request)
    {
        string result = "";
        using (Stream req = request.InputStream)
        {
            req.Seek(0, System.IO.SeekOrigin.Begin);
            result = new StreamReader(req).ReadToEnd();
        }
        return JsonConvert.DeserializeObject<T>(result);
    }

0

如果您要从控制器返回视图,并且想在使用cshtml中的json编码时增加视图包数据的长度,则可以将此代码放入cshtml中

@{
    var jss = new System.Web.Script.Serialization.JavaScriptSerializer();
    jss.MaxJsonLength = Int32.MaxValue;
    var userInfoJson = jss.Serialize(ViewBag.ActionObj);
}

var dataJsonOnActionGrid1 = @Html.Raw(userInfoJson);

现在, dataJsonOnActionGrid1 将可在js页面上访问,您将获得正确的结果。

谢谢

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.