MaxJsonLength ข้อยกเว้นใน ASP.NET MVC ระหว่าง JavaScriptSerializer


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
@ มัทเทโอแน่ใจเหรอ? นี่เป็นคำถามเก่าและฉันจำไม่ได้ แต่เห็นได้ชัดว่าฉันแท็กเป็น MVC3 น่าเสียดายที่ฉันไม่สามารถดูเวอร์ชัน / วันที่เมื่อได้รับการแก้ไข / ปิด: aspnet.codeplex.com/workitem/3436
Martin Buberl

1
ได้เลยฉันกำลังทำงานกับ MVC 3 และใช้งานได้ และโชคดีเพราะใน MVC 3 คุณไม่มีคุณสมบัติ "MaxJsonLength" ที่อ้างถึงในคำตอบที่ยอมรับ
MatteoSp

คำตอบ:


228

ดูเหมือนว่าสิ่งนี้ได้รับการแก้ไขแล้วใน MVC4

คุณสามารถทำได้ซึ่งได้ผลดีสำหรับฉัน:

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

ฉันกำลังตั้งค่าสตริง json ในคุณสมบัติ ViewBag.MyJsonString แต่ได้รับข้อผิดพลาดเดียวกันในมุมมองของฉันที่รันไทม์บนบรรทัดจาวาสคริปต์ต่อไปนี้: var myJsonObj = @ Html.Raw (Json.Encode (ViewBag.MyJsonString));
Faisal Mq

1
เฮ้ @orionedvards, @ GG, @ MartinBuberl ฉันกำลังประสบปัญหา maxJson เหมือนกัน แต่เมื่อโพสต์ข้อมูลไปยังคอนโทรลเลอร์ฉันจะจัดการสิ่งนี้ได้อย่างไรฉันใช้เวลาค้นหาเรื่องนี้มากความช่วยเหลือใด ๆ จะขอบคุณ
katmanco

ในกรณีของฉันไม่ได้ผลเพราะฉันต้องตั้งค่า MaxJsonLength ก่อน json ต่อเนื่องคอลเลกชัน
Césarเลออน

ในกรณีของฉันใช้งานได้ดีฉันต้องใช้มันเนื่องจาก "IMAGES" ใน datatable เพื่อนำเสนอสำหรับผู้ใช้ขั้นสุดท้าย ถ้าไม่มีก็ผิดพลาดโดยไม่มีข้อความที่เข้าใจได้
Mauro Candido

33

นอกจากนี้คุณยังสามารถใช้ContentResultเป็นแนะนำที่นี่แทน JsonResultsubclassing

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

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

2
ในกรณีของฉันการทำงานกับแอปพลิเคชันที่ถูกทิ้งวิธีนี้ได้ผลดีที่สุดสำหรับฉัน บันทึกการใช้งาน jsonresult ขอบคุณ!
Christo


22

ไม่จำเป็นต้องมีคลาสที่กำหนดเอง นี่คือสิ่งที่จำเป็น:

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

ซึ่งResultเป็นข้อมูลที่คุณต้องการที่จะเป็นอันดับ


ข้อผิดพลาด 137 'System.Web.Mvc.JsonResult' ไม่ได้มีความหมายสำหรับ 'MaxJsonLength'
ปั๊ก

สิ่งนี้ใช้ได้ผลสำหรับฉัน แต่ก็ยังจำเป็นต้องเพิ่ม: JsonRequestBehavior = JsonRequestBehavior.A 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 ของคุณเฉพาะฟิลด์ที่คุณต้องการ

ตัวอย่าง. ลองนึกภาพว่าคุณมี Model ที่มี Id, Name, Phone และPicture (byte array)และต้องโหลดจาก json ลงในรายการที่เลือก

คำถาม LINQ:

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

ปัญหาที่นี่คือ " เลือกคุณ " ที่ได้รับฟิลด์ทั้งหมด ดังนั้นหากคุณมีภาพขนาดใหญ่ booomm

วิธีแก้? ง่ายมาก

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 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แต่มีการเพิ่มค่า appsetting 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;
    }
}

ขอบคุณมาก !
larizzatg

1

ข้างต้นไม่ได้ผลสำหรับฉันจนกว่าฉันจะเปลี่ยนการกระทำเป็น[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;
     ....
    }

ระบบแสดงข้อยกเว้นเช่นนี้ "Error during serialization หรือ deserialization โดยใช้ JSON JavaScriptSerializer ความยาวของสตริงเกินค่าที่ตั้งไว้ในคุณสมบัติ maxJsonLength ชื่อพารามิเตอร์: input"

การเปลี่ยนการตั้งค่า Web.config เท่านั้นไม่เพียงพอที่จะช่วยในกรณีนี้ คุณสามารถลบล้างซีเรียลไลเซอร์ mvc json เพิ่มเติมเพื่อรองรับขนาดโมเดลข้อมูลขนาดใหญ่หรือโมเดล deserialize ด้วยตนเองจากคำขอ วิธีการควบคุมของคุณกลายเป็น:

   [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 และคุณจะได้รับผลลัพธ์ที่เหมาะสม

ขอบคุณ

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.