Исключение MaxJsonLength в ASP.NET MVC во время JavaScriptSerializer - PullRequest
112 голосов
/ 17 апреля 2011

В одном из действий моего контроллера я возвращаю очень большое JsonResult для заполнения сетки.

Я получаю следующее InvalidOperationException исключение:

Ошибка во времясериализация или десериализация с использованием JSON JavaScriptSerializer.Длина строки превышает значение, установленное в свойстве maxJsonLength.

Установка для свойства maxJsonLength в web.config более высокого значения, к сожалению, не дает никакого эффекта.

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

Я не хочу передавать его обратно в виде строки, как указано в этом ТАКОМ ответе.

В своем исследовании я натолкнулся на это сообщение в блогегде рекомендуется написать собственный ActionResult (например, LargeJsonResult : JsonResult), чтобы обойти это поведение.

Это единственное решение?
Это ошибка в ASP.NET MVC?
AmЯ что-то упустил?

Любая помощь будет наиболее ценной.

Ответы [ 14 ]

203 голосов
/ 20 декабря 2012

Похоже, это было исправлено в MVC4.

Вы можете сделать это, что хорошо сработало для меня:

public ActionResult SomeControllerAction()
{
  var jsonResult = Json(veryLargeCollection, JsonRequestBehavior.AllowGet);
  jsonResult.MaxJsonLength = int.MaxValue;
  return jsonResult;
}
32 голосов
/ 29 марта 2012

Вы также можете использовать ContentResult в качестве , предложенного здесь вместо подкласса JsonResult.

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

return new ContentResult()
{
    Content = serializer.Serialize(data),
    ContentType = "application/json",
};
24 голосов
/ 17 апреля 2011

К сожалению, настройка web.config игнорируется реализацией JsonResult по умолчанию . Поэтому я думаю, вам нужно будет реализовать пользовательский результат json, чтобы преодолеть эту проблему.

20 голосов
/ 08 апреля 2013

Нет необходимости в пользовательском классе.Это все, что нужно:

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

, где Result - это те данные, которые вы хотите сериализовать.

5 голосов
/ 06 февраля 2015

Если для генерации строки json используется Json.NET , не нужно устанавливать значение MaxJsonLength.

return new ContentResult()
{
    Content = Newtonsoft.Json.JsonConvert.SerializeObject(data),
    ContentType = "application/json",
};
4 голосов
/ 16 июня 2016

Я решил проблему, перейдя по этой ссылке

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());
    }
2 голосов
/ 03 июня 2018

Альтернативный ASP.NET MVC 5 Fix:

В моем случае во время запроса произошла ошибка. Лучший подход в моем сценарии - это изменение фактического 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;
    }
}
2 голосов
/ 24 сентября 2014

Вы можете попытаться определить в своем выражении LINQ только те поля, которые вам понадобятся.

Пример.Представьте, что у вас есть Модель с Id, Name, Phone и Picture (байтовый массив) и вам нужно загрузить из json в список выбора.

LINQ Query:

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

Проблема здесь в том, что " выберите u ", которые получают все поля.Так что, если у вас есть большие картинки, booomm.

Как решить?очень, очень просто.

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

Рекомендуется выбирать только те поля, которые вы будете использовать.

Запомните.Это простой совет, но он может помочь многим разработчикам ASP.NET MVC.

1 голос
/ 04 июля 2018

Ничто из вышеперечисленного не сработало для меня, пока я не сменил действие на [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...");
    }
});
0 голосов
/ 01 апреля 2019

Вы можете поместить этот код в 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, и вы получите правильный результат.

Спасибо

...