ฉันสามารถตั้งค่าความยาวไม่ จำกัด สำหรับ maxJsonLength ใน web.config ได้หรือไม่


663

ฉันกำลังใช้คุณสมบัติการเติมข้อความอัตโนมัติของ jQuery เมื่อฉันพยายามเรียกดูรายการมากกว่า 17,000 รายการ (แต่ละรายการจะมีความยาวไม่เกิน 10 ถ่าน) มันยาวเกินความเป็นจริงและเกิดข้อผิดพลาด:

ข้อมูลข้อยกเว้น:
ประเภทข้อยกเว้น: InvalidOperationException
ข้อความข้อยกเว้น: ข้อผิดพลาดระหว่างการทำให้เป็นอนุกรมหรือการดีซีเรียลไลเซชันโดยใช้ JSON JavaScriptSerializer ความยาวของสตริงเกินกว่าค่าที่ตั้งไว้ในคุณสมบัติ maxJsonLength

ฉันสามารถตั้งค่าความยาวได้ไม่ จำกัดmaxJsonLengthในweb.configหรือไม่ ถ้าไม่ความยาวสูงสุดที่ฉันสามารถตั้งได้คือเท่าไหร่


1
สิ่งที่จะพูดถึงซึ่งอาจจะค่อนข้างชัดเจนดังนั้นโปรดยกโทษให้ฉันถ้าคุณคิดแล้วถ้ามัน; สตริง Json ยังมีวงเล็บปีกการอบแต่ละเรกคอร์ดเครื่องหมายอัญประกาศรอบแต่ละชื่อฟิลด์ [และค่า], รวมถึงชื่อฟิลด์และค่า ดังนั้นอาจเป็นประโยชน์ในการตั้งชื่อเขตข้อมูลเป็นอักขระเดียวและตรวจสอบให้แน่ใจว่าหากค่าไม่ใช่สตริงคุณจะต้องตั้งค่าประเภทฟิลด์ให้ถูกต้องดังนั้นจึงไม่มีเครื่องหมายคำพูด
MichaelJTaylor

คำตอบ:


719

หมายเหตุ:คำตอบนี้ใช้ได้กับเว็บเซอร์วิสเท่านั้นหากคุณส่งคืน JSON จากวิธีควบคุมตรวจสอบให้แน่ใจว่าคุณอ่านคำตอบ SO ด้านล่างนี้ด้วย: https://stackoverflow.com/a/7207539/1246870


MaxJsonLengthคุณสมบัติไม่สามารถไม่ จำกัด เป็นจำนวนเต็มคุณสมบัติที่เริ่มต้นที่ 102400 (100k)

คุณสามารถตั้งค่าMaxJsonLengthคุณสมบัติบน web.config ของคุณ:

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

153
มันเป็นจำนวนเต็มดังนั้นค่าสูงสุดที่คุณสามารถตั้งได้คือ: 2147483644
David Espart

57
@despart: คุณหมายถึง 2 147 483 647.
Dercsár

6
@ kmcc049, IMO ค่าไม่ผิดเพราะถ้าคุณดูคำถาม OP จะไม่ถามว่า "ค่าเริ่มต้นของ maxJsonLength คืออะไร" (BTW คำตอบที่ได้รับคะแนนรองอันดับสองคือตอบคำถามนี้ผิดคำถาม) เขาพยายามตั้งค่าคุณสมบัตินี้เป็น "ไม่ จำกัด " แต่เนื่องจากเป็นจำนวนเต็มค่าสูงสุดที่เป็นไปได้คือ2147483647@depsart และ @ Descárชี้ให้เห็น
CMS

11
ยอดเยี่ยม แต่โปรดทราบคำตอบของ @David Murdoch ด้านล่างหากคุณมีปัญหานี้เมื่อใช้งาน MVC return Json()หรือบางสิ่งบางอย่าง
BritishDeveloper

3
@ Dercsár: ประเด็นคืออะไร? 2147483644 เป็นจำนวนเต็มที่ใหญ่ที่สุดหารด้วย 1024 ได้อย่างสมบูรณ์
naveen

461

หากคุณใช้ MVC 4โปรดตรวจสอบคำตอบนี้เช่นกัน


หากคุณยังคงได้รับข้อผิดพลาด:

  • หลังจากตั้งค่า maxJsonLengthคุณสมบัติเป็นค่าสูงสุดใน web.config
  • และคุณรู้ว่าความยาวของข้อมูลของคุณน้อยกว่าค่านี้
  • และคุณไม่ได้ใช้วิธีการบริการเว็บสำหรับการทำให้เป็นอนุกรม JavaScript

ปัญหาของคุณมีแนวโน้มว่า:

ค่าของคุณสมบัติ MaxJsonLength ใช้เฉพาะกับอินสแตนซ์ JavaScriptSerializer ภายในที่ใช้โดยเลเยอร์การสื่อสารแบบอะซิงโครนัสเพื่อเรียกใช้วิธีการบริการเว็บ ( MSDN: คุณสมบัติ ScriptingJsonSerializationSection.MaxJsonLength )

โดยทั่วไป "ภายใน" จะJavaScriptSerializerเคารพคุณค่าของmaxJsonLengthเมื่อถูกเรียกจากวิธีการบนเว็บ การใช้งานโดยตรงของJavaScriptSerializer(หรือใช้ผ่าน MVC action-method / Controller) ไม่เกี่ยวข้องกับmaxJsonLengthคุณสมบัติอย่างน้อยไม่ได้มาจากsystemWebExtensions.scripting.webServices.jsonSerializationส่วนของ web.config

คุณสามารถทำสิ่งต่อไปนี้ภายในคอนโทรลเลอร์ของคุณ (หรือที่ใดก็ได้):

var serializer = new JavaScriptSerializer();

// For simplicity just use Int32's max value.
// You could always read the value from the config section mentioned above.
serializer.MaxJsonLength = Int32.MaxValue;

var resultData = new { Value = "foo", Text = "var" };
var result = new ContentResult{
    Content = serializer.Serialize(resultData),
    ContentType = "application/json"
};
return result;

คำตอบนี้คือความหมายของฉันคำตอบฟอรั่ม asp.net นี้


5
คำตอบของคุณมีประโยชน์จริง ๆ เพราะฉันใช้Json()วิธีผลการดำเนินการใน asp.net mvc
jessegavin

3
ใช่ฉันเป็น Json () ต้องทนทุกข์เช่นกัน ขอบคุณ!
พัฒนาชาวอังกฤษ

3
แม้ว่ามันจะถูกต้องสมบูรณ์และสมควรได้รับตำแหน่ง แต่นี่เป็นหนึ่งในคำถามที่คุ้มค่าที่จะอ่านผ่านคำตอบที่ดีที่สุด :) ขอบคุณ!
Nigel

3
หากคุณใช้ MVC4 โปรดดู @fanisch คำตอบด้วย
เบเยอร์

4
Deserialization ฉันพบข้อผิดพลาดนี้ที่การโยงโมเดลของแอ็คชัน
guogangj

345

ใน MVC 4 คุณสามารถทำได้:

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
    };
}

ในตัวควบคุมของคุณ

ส่วนที่เพิ่มเข้าไป:

สำหรับทุกคนที่สับสนด้วยพารามิเตอร์ที่คุณต้องระบุการโทรอาจมีลักษณะเช่นนี้:

Json(
    new {
        field1 = true,
        field2 = "value"
        },
    "application/json",
    Encoding.UTF8,
    JsonRequestBehavior.AllowGet
);

6
ฉันสามารถยืนยันได้ว่าการทำงานดังกล่าวมีเสน่ห์ใน MVC 4 ขอบคุณแฟนเพจ
เบเยอร์

9
ฉันสามารถยืนยันได้เช่นกัน การวางรหัสนี้ไว้ในคอนโทรลเลอร์พื้นฐานเป็นวิธีที่สะอาดที่สุดที่เสนอ
รัฐสภา

15
สิ่งนี้ยังทำงานได้โดยเพียงแค่เพิ่ม "MaxJsonLength = Int32.MaxValue" ให้กับผลการดำเนินการแต่ละรายการ ในกรณีที่การเปลี่ยนแปลงไม่เป็นที่ต้องการตัวควบคุมหรือโครงการกว้าง
Hypnovirus

3
นี่คือคำตอบที่ดีที่สุด MaxJsonLength สามารถกำหนดค่าสำหรับแต่ละคอนโทรลเลอร์
เหลียง

3
คำเตือน: วิธีนี้จะปิดการใช้งานการบีบอัด (ถ้ามีการร้องขอ) ของการตอบสนอง เพิ่มตัวกรองนี้ในการกระทำของคุณ: stackoverflow.com/questions/3802107/ …
Gorgi Rankovski

60

คุณสามารถกำหนดค่าความยาวสูงสุดสำหรับคำขอ json ในไฟล์ web.config ของคุณ:

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

ค่าเริ่มต้นสำหรับ maxJsonLength คือ102400 สำหรับรายละเอียดเพิ่มเติมดูที่หน้า MSDN นี้: http://msdn.microsoft.com/en-us/library/bb763183.aspx


1
ค่าที่เก็บไว้ในจำนวนเต็มนี้หมายถึงอะไร นี่เป็นตัวละครที่นับได้ไหม? ฉันเดาว่าฉันกำลังถามอะไรอยู่ทำไมจึงใช้จำนวนเต็ม? ขอบคุณ!
eaglei22

@ eaglei22 ตัวเลขแสดงจำนวนไบต์ที่สามารถใช้สำหรับ maxJsonLength ดังที่ M4N กล่าวถึง 102400 เป็นค่าเริ่มต้น (100KB)
Jacob Plonke

สิ่งนี้ไม่ได้ผลสำหรับฉันและฉันไม่ได้ใช้บริการเว็บ
kalai

42

หากคุณยังคงได้รับข้อผิดพลาดหลังจากการตั้งค่า web.config ดังนี้:

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

ฉันแก้ไขมันโดยทำตาม:

   public ActionResult/JsonResult getData()
   {
      var jsonResult = Json(superlargedata, JsonRequestBehavior.AllowGet);
      jsonResult.MaxJsonLength = int.MaxValue;
      return jsonResult;
    }

ฉันหวังว่านี่จะช่วยได้


2
การตั้งค่า maxJsonLength ใน web.config เป็น unnesseary การตั้งค่า jsonResult.MaxJsonLength ควรพอเพียง (อย่างน้อยก็ทำเพื่อฉัน (MVC5))
hormberg

นี่เป็นสิ่งที่ดีเพราะมันไม่ใช่การเปลี่ยนแปลงระดับโลก
rob_james

40

ฉันมีปัญหานี้ในเว็บฟอร์ม ASP.NET มันไม่สนใจการตั้งค่าไฟล์ web.config อย่างสมบูรณ์ดังนั้นฉันจึงทำสิ่งนี้:

        JavaScriptSerializer serializer = new JavaScriptSerializer();

        serializer.MaxJsonLength = Int32.MaxValue; 

        return serializer.Serialize(response);

แน่นอนว่าโดยรวมนี่เป็นการฝึกฝนที่แย่มาก หากคุณกำลังส่งข้อมูลจำนวนมากในการโทรจากเว็บเซอร์วิสคุณควรดูวิธีอื่น


1
มันทำงานให้คุณหรือไม่ คุณวางรหัสนี้ที่ไหน
user1012598

ปัญหาของเราคือเนื่องจากเรามี textarea ที่อนุญาตสำหรับ HTML และผู้คนฝังรูปภาพเป็น HTML ซึ่งทำให้รายการมีขนาดใหญ่มากและ JSON serializer ล้มเหลว ผมคิดว่าถ้ามันสามารถทำได้ผู้ใช้จะทำมัน ...
Marko

โปรดอธิบายว่าเราควรวางรหัสนี้ไว้ที่ไหน ... @Flea
Koray Durudogan

@KorayDurudogan - ฉันใส่สิ่งนี้ในวิธี Ajax ที่กลับมาตอบสนองดังนั้นในตัวควบคุมของฉัน หวังว่าจะช่วย!
หมัด

ฉันไม่ได้ท้าทายการตอบสนองของคุณ แต่พยายามที่จะได้แนวคิดที่ดีกว่าเกี่ยวกับวิธีการที่ดีกว่า ฉันมีแบบสอบถามที่ขึ้นอยู่กับเกณฑ์ของผู้ใช้ที่จะกำหนดขนาดผลลัพธ์ ฉันคืน JsonResult มันจะสำคัญไหมถ้าฉันคืนไฟล์ excel?
eaglei22

22

ฉันซ่อมมัน.

//your Json data here
string json_object="........";
JavaScriptSerializer jsJson = new JavaScriptSerializer();
jsJson.MaxJsonLength = 2147483644;
MyClass obj = jsJson.Deserialize<MyClass>(json_object);

มันใช้งานได้ดีมาก


! น่ากลัว นี่เป็นทางออกเดียวที่ทำงานให้ฉันและดีกว่านี้อีกต่อไปเพราะมันไม่ใช่การเปลี่ยนแปลงระดับโลก ขอบคุณ!
Sealer_05

20

ฉันทำตามคำตอบของร่องรอยและได้แก้ปัญหานี้:

เมื่อฉันต้องการโพสต์ json ขนาดใหญ่ไปยังการดำเนินการในคอนโทรลเลอร์ฉันจะได้รับ "ข้อผิดพลาดระหว่างการดีซีเรียลไลเซชันโดยใช้ JSON JavaScriptSerializer ความยาวของสตริงเกินกว่าค่าที่ตั้งไว้ในคุณสมบัติ maxJsonLength \ r \ n พารามิเตอร์ชื่อ: อินพุต ผู้ให้บริการมูลค่า "

สิ่งที่ฉันทำคือสร้าง ValueProviderFactory ใหม่ LargeJsonValueProviderFactory และตั้งค่า MaxJsonLength = Int32.MaxValue ในเมธอด GetDeserializedObject

public sealed class LargeJsonValueProviderFactory : ValueProviderFactory
{
private static void AddToBackingStore(LargeJsonValueProviderFactory.EntryLimitedDictionary backingStore, string prefix, object value)
{
    IDictionary<string, object> dictionary = value as IDictionary<string, object>;
    if (dictionary != null)
    {
        foreach (KeyValuePair<string, object> keyValuePair in (IEnumerable<KeyValuePair<string, object>>) dictionary)
            LargeJsonValueProviderFactory.AddToBackingStore(backingStore, LargeJsonValueProviderFactory.MakePropertyKey(prefix, keyValuePair.Key), keyValuePair.Value);
    }
    else
    {
        IList list = value as IList;
        if (list != null)
        {
            for (int index = 0; index < list.Count; ++index)
                LargeJsonValueProviderFactory.AddToBackingStore(backingStore, LargeJsonValueProviderFactory.MakeArrayKey(prefix, index), list[index]);
        }
        else
            backingStore.Add(prefix, value);
    }
}

private static object GetDeserializedObject(ControllerContext controllerContext)
{
    if (!controllerContext.HttpContext.Request.ContentType.StartsWith("application/json", StringComparison.OrdinalIgnoreCase))
        return (object) null;
    string end = new StreamReader(controllerContext.HttpContext.Request.InputStream).ReadToEnd();
    if (string.IsNullOrEmpty(end))
        return (object) null;

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

    return serializer.DeserializeObject(end);
}

/// <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 = LargeJsonValueProviderFactory.GetDeserializedObject(controllerContext);
    if (deserializedObject == null)
        return (IValueProvider) null;
    Dictionary<string, object> dictionary = new Dictionary<string, object>((IEqualityComparer<string>) StringComparer.OrdinalIgnoreCase);
    LargeJsonValueProviderFactory.AddToBackingStore(new LargeJsonValueProviderFactory.EntryLimitedDictionary((IDictionary<string, object>) dictionary), string.Empty, deserializedObject);
    return (IValueProvider) new DictionaryValueProvider<object>((IDictionary<string, object>) dictionary, CultureInfo.CurrentCulture);
}

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

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

private class EntryLimitedDictionary
{
    private static int _maximumDepth = LargeJsonValueProviderFactory.EntryLimitedDictionary.GetMaximumDepth();
    private readonly IDictionary<string, object> _innerDictionary;
    private int _itemCount;

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

    public void Add(string key, object value)
    {
        if (++this._itemCount > LargeJsonValueProviderFactory.EntryLimitedDictionary._maximumDepth)
            throw new InvalidOperationException("JsonValueProviderFactory_RequestTooLarge");
        this._innerDictionary.Add(key, value);
    }

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

จากนั้นในเมธอด Application_Start จาก Global.asax.cs ให้แทนที่ ValueProviderFactory ด้วยอันใหม่:

protected void Application_Start()
{
    ...

    //Add LargeJsonValueProviderFactory
    ValueProviderFactory jsonFactory = null;
    foreach (var factory in ValueProviderFactories.Factories)
    {
        if (factory.GetType().FullName == "System.Web.Mvc.JsonValueProviderFactory")
        {
            jsonFactory = factory;
            break;
        }
    }

    if (jsonFactory != null)
    {
        ValueProviderFactories.Factories.Remove(jsonFactory);
    }

    var largeJsonValueProviderFactory = new LargeJsonValueProviderFactory();
    ValueProviderFactories.Factories.Add(largeJsonValueProviderFactory);
}

1
ฉันทำทุกสิ่งเท่าที่ทำได้ แต่คำตอบของคุณช่วยชีวิตฉันไว้คำตอบนี้ควรได้รับการยอมรับ
Muhammad Waqas Aziz

ด้วยรหัสนี้เราสามารถแทนที่ตัวควบคุม MVC สูงสุด json Deserializetion ขีด จำกัด ที่ 4 mb แต่มีวิธีที่จะแทนที่ตัวควบคุม web-api ขีด จำกัด สูงสุดของ dessonialetetet json
Muhammad Waqas Aziz

17

หากหลังจากใช้งานการเพิ่มข้างต้นลงใน web.config ของคุณคุณจะได้รับ“ ส่วนการกำหนดค่าที่ไม่รู้จักระบบ. web.web.extensions” เกิดข้อผิดพลาดจากนั้นลองเพิ่มลงในเว็บของคุณกำหนดค่าใน<ConfigSections>ส่วน:

            <sectionGroup name="system.web.extensions" type="System.Web.Extensions">
              <sectionGroup name="scripting" type="System.Web.Extensions">
                    <sectionGroup name="webServices" type="System.Web.Extensions">
                          <section name="jsonSerialization" type="System.Web.Extensions"/>
                    </sectionGroup>
              </sectionGroup>
        </sectionGroup>

4
ฉันมีปัญหานี้ อย่างไรก็ตามคำตอบนี้ไม่ได้ผลสำหรับฉัน แทนที่จะเพิ่มองค์ประกอบ <sectionGroup> ที่อธิบายไว้ที่นี่ฉันเพิ่งย้ายบล็อก <system.web.extensions> ที่เพิ่งเพิ่มไปยังส่วนท้ายสุดของ web.config ... ทันทีก่อน </configuration> จากนั้นก็ใช้งานได้
ClearCloud8

สิ่งนี้ช่วยได้ แต่ในสถานการณ์ของฉันฉันต้องเปลี่ยนบรรทัดที่สี่ของคุณ<section name="jsonSerialization" type="System.Web.Configuration.ScriptingJsonSerializationSection, System.Web.Extensions, Version=3.5.0.0, Culture=neutral, PublicKeyToken=31BF3856AD364E35" requirePermission="false" allowDefinition="Everywhere"/>ตามที่เห็นในหน้านี้: forums.asp.net/t/1446510.aspx/1
Nathan

@ ClearCloud8 รับความคิดเห็นนั้นแพร่กระจายไปทั่วหน้านี้ทันที
แจ็ค Nutkins

11

คุณสามารถเขียนบรรทัดนี้เป็นคอนโทรลเลอร์

json.MaxJsonLength = 2147483644;

คุณยังสามารถเขียนบรรทัดนี้เป็น web.config

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

`

หากต้องการอยู่ด้านปลอดภัยให้ใช้ทั้งสองอย่าง


10

หากคุณได้รับข้อผิดพลาดนี้จากMiniProfilerใน MVC คุณสามารถเพิ่มค่าได้โดยการตั้งค่าคุณสมบัติMiniProfiler.Settings.MaxJsonResponseSizeเป็นค่าที่ต้องการ โดยค่าเริ่มต้นเครื่องมือนี้ดูเหมือนจะไม่สนใจค่าที่ตั้งไว้ในการกำหนดค่า

MiniProfiler.Settings.MaxJsonResponseSize = 104857600;

มารยาทMVC-มินิ Profiler


10

เพียงตั้งค่า MaxJsonLength proprty ในวิธีการของ MVC

JsonResult json= Json(classObject, JsonRequestBehavior.AllowGet);
json.MaxJsonLength = int.MaxValue;
return json;

9

ฉันแนะนำให้ตั้งค่าเป็น Int32.MaxValue

JavaScriptSerializer serializer = new JavaScriptSerializer();
serializer.MaxJsonLength = Int32.MaxValue;

9

เวทมนตร์เกี่ยวกับคุณลักษณะบางอย่าง?

[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, Inherited = true, AllowMultiple = false)]
public class MaxJsonSizeAttribute : ActionFilterAttribute
{
    // Default: 10 MB worth of one byte chars
    private int maxLength = 10 * 1024 * 1024;

    public int MaxLength
    {
        set
        {
            if (value < 0) throw new ArgumentOutOfRangeException("value", "Value must be at least 0.");

            maxLength = value;
        }
        get { return maxLength; }
    }

    public override void OnActionExecuted(ActionExecutedContext filterContext)
    {
        JsonResult json = filterContext.Result as JsonResult;
        if (json != null)
        {
            if (maxLength == 0)
            {
                json.MaxJsonLength = int.MaxValue;
            }
            else
            {
                json.MaxJsonLength = maxLength;
            }
        }
    }
}

จากนั้นคุณสามารถใช้ทั่วโลกโดยใช้การกำหนดค่าตัวกรองทั่วโลกหรือคอนโทรลเลอร์ / การกระทำที่ชาญฉลาด


คำตอบที่ดี ใช้แอตทริบิวต์ที่กำหนดเองได้ดี หากคุณสงสัยว่ามีเหตุผลเฉพาะทางเทคนิคหรือไม่ที่คุณตั้งค่าเริ่มต้นเป็น 10 MB สำหรับหนึ่งไบต์แทนที่จะเป็น Max (int.MaxValue)
Josh

@ Josh ไม่ไม่มีเหตุผลพิเศษสำหรับเรื่องนั้น
Balázs

5

คำถามคือคุณต้องการส่งคืนระเบียน 17k จริงหรือไม่ คุณวางแผนที่จะจัดการกับข้อมูลทั้งหมดในเบราว์เซอร์อย่างไร? ผู้ใช้จะไม่เลื่อนผ่าน 17,000 แถว

วิธีที่ดีกว่าคือการดึงเฉพาะระเบียน "น้อยที่สุด" และโหลดเพิ่มเติมตามที่ต้องการ


1
รายการเริ่มต้นจาก json จะให้บันทึก 17k แต่คุณสมบัติการเติมข้อความอัตโนมัติจะแสดงเฉพาะระเบียนที่ตรงกับตัวละครที่ผู้ใช้พิมพ์ดังนั้นจึงไม่จำเป็นต้องเลื่อนรายการเพิ่มเติม ดังนั้นสิ่งที่ฉันต้องการคือการตั้งค่าความยาวไม่ จำกัด สำหรับ maxJsonLength ซึ่งสามารถทำให้เป็นอนุกรมข้อมูล 17k
Prasad

6
คุณสามารถใช้การกรองเซิร์ฟเวอร์และฝั่งไคลเอ็นต์ร่วมกัน อาจเป็นการยากที่จะกรองข้อมูลทั้งหมดในฝั่งไคลเอ็นต์ไม่ต้องพูดถึงเวลาแฝงของเครือข่าย
Chetan Sastry

1
เมื่อมาถึงที่ปัญหาเดียวกันในขณะนี้กลับฉันเลือกที่จะใช้ตัวจัดการ "onsearch" สำหรับการเติมข้อความอัตโนมัติและให้บริการเว็บผ่านข้อความ "ค้นหา" และทำแบบสอบถาม Top10 โดยใช้เกณฑ์การค้นหาเป็นตัวกรอง นี่หมายถึงคำขอ ajax ส่วนตัวที่มากขึ้นเพียงแค่รับรายการทั้งหมดในการโหลดหน้าเว็บ แต่ก็หมายความว่าคำขอ / การตอบกลับทั้งหมดมีขนาดเล็กกว่ามาก
Mike U

3

คุณสามารถตั้งค่าในการกำหนดค่าตามที่คนอื่นพูดหรือคุณสามารถตั้งค่าในแต่ละอินสแตนซ์ของ serializer เช่น:

var js = new JavaScriptSerializer() { MaxJsonLength = int.MaxValue };

3

สำหรับผู้ที่มีปัญหากับใน MVC3 กับ JSON ที่มีการดีซีเรียลไลซ์อัตโนมัติสำหรับเครื่องผูกโมเดลและมีขนาดใหญ่เกินไปนี่คือวิธีแก้ปัญหา

  1. คัดลอกรหัสสำหรับคลาส JsonValueProviderFactory จากซอร์สโค้ด MVC3 ไปยังคลาสใหม่
  2. เพิ่มบรรทัดเพื่อเปลี่ยนความยาว JSON สูงสุดก่อนที่วัตถุจะทำการดีซีเรียลไลซ์
  3. แทนที่คลาส JsonValueProviderFactory ด้วยคลาสใหม่ที่ถูกปรับเปลี่ยนของคุณ

ขอบคุณhttp://blog.naver.com/techshare/100145191355และhttps://gist.github.com/DalSoft/1588818สำหรับการชี้ให้ฉันในทิศทางที่ถูกต้องสำหรับวิธีการทำเช่นนี้ ลิงค์สุดท้ายในเว็บไซต์แรกมีรหัสที่มาแบบเต็มสำหรับการแก้ปัญหา


3

หากคุณพบปัญหาประเภทนี้ในมุมมองคุณสามารถใช้วิธีการด้านล่างเพื่อแก้ไขปัญหาดังกล่าว นี่คือแพ็คเกจIused Newtonsoft

@using Newtonsoft.Json
<script type="text/javascript">
    var partData = @Html.Raw(JsonConvert.SerializeObject(ViewBag.Part));
</script>

นี่หมายความว่าฉันไม่ต้องกังวลกับความยาวสูงสุดถ้าฉันใช้ Json.NET? ฉันไม่คิดว่าจะมีวิธีการตั้งค่าความยาวสูงสุดใน Json.NET ดังนั้นฉันจึงหวังว่ามันจะออกมานอกกรอบ
kimbaudi

1
คำตอบที่ยอดเยี่ยมขอบคุณ! สิ่งนี้ยังทำงานได้เมื่อฉันพยายามโหลดวัตถุ
user1299379


2

ปรากฏว่าไม่มีค่า "ไม่ จำกัด " ค่าเริ่มต้นคือ 2097152 อักขระซึ่งเทียบเท่ากับข้อมูลสตริง Unicode 4 MB

อย่างที่ทราบแล้วบันทึก 17,000 รายการใช้งานได้ยากในเบราว์เซอร์ หากคุณกำลังแสดงมุมมองรวมอาจมีประสิทธิภาพมากกว่าในการรวมบนเซิร์ฟเวอร์และถ่ายโอนเฉพาะบทสรุปในเบราว์เซอร์ ตัวอย่างเช่นพิจารณาเบราว์เซอร์ระบบไฟล์เราจะเห็นเฉพาะส่วนบนสุดของต้นไม้จากนั้นปล่อยคำขอเพิ่มเติมเมื่อเราเจาะลึกลงไป จำนวนเร็กคอร์ดที่ส่งคืนในแต่ละคำร้องขอมีขนาดค่อนข้างเล็ก การนำเสนอมุมมองแบบต้นไม้สามารถทำงานได้ดีสำหรับชุดผลลัพธ์ขนาดใหญ่


3
ค่อนข้างแปลกเริ่มต้นในรหัส (ใหม่ JavaScriptSerializer ()) MaxJsonLength คือ 2097152 ไบต์ แต่บริการเว็บ ResponseFormatJson เป็น 102400 ไบต์เว้นแต่ตั้งอย่างชัดเจน
ปล้น

2

เพิ่งเจอกับสิ่งนี้ ฉันได้รับบันทึกมากกว่า 6,000 รายการ เพิ่งตัดสินใจฉันจะทำเพจจิ้ง เช่นเดียวกับในฉันยอมรับหมายเลขหน้าในจุดสิ้นสุด MVC JsonResult ของฉันซึ่งมีค่าเริ่มต้นเป็น 0 ดังนั้นจึงไม่จำเป็นเช่น:

public JsonResult MyObjects(int pageNumber = 0)

จากนั้นแทนที่จะพูดว่า:

return Json(_repository.MyObjects.ToList(), JsonRequestBehavior.AllowGet);

ฉันพูด:

return Json(_repository.MyObjects.OrderBy(obj => obj.ID).Skip(1000 * pageNumber).Take(1000).ToList(), JsonRequestBehavior.AllowGet);

มันง่ายมาก จากนั้นใน JavaScript แทนที่จะเป็น:

function myAJAXCallback(items) {
    // Do stuff here
}

ฉันพูดแทน:

var pageNumber = 0;
function myAJAXCallback(items) {
    if(items.length == 1000)
        // Call same endpoint but add this to the end: '?pageNumber=' + ++pageNumber
    }
    // Do stuff here
}

และผนวกบันทึกของคุณกับสิ่งที่คุณทำกับพวกเขาในตอนแรก หรือรอจนกว่าการโทรทั้งหมดจะเสร็จสิ้น


2

ฉันแก้ไขปัญหาการเพิ่มรหัสนี้:

String confString = HttpContext.Current.Request.ApplicationPath.ToString();
Configuration conf = WebConfigurationManager.OpenWebConfiguration(confString);
ScriptingJsonSerializationSection section = (ScriptingJsonSerializationSection)conf.GetSection("system.web.extensions/scripting/webServices/jsonSerialization");
section.MaxJsonLength = 6553600;
conf.Save();

ดูเหมือนว่าจะเป็นวิธีการแฮ็ก แต่วิธีการที่น่าสนใจโดยไม่คำนึงถึง ฉันพบว่ามีประโยชน์ขอบคุณ! สำหรับฉันในคอนโทรลเลอร์ apsnet mvc 5 ฉันต้องลบ 'ปัจจุบัน' ออกจากเนมสเปซ ฉันทำการปรับแต่งสองสามครั้ง:string confString = HttpContext.Request.ApplicationPath.ToString(); var conf = System.Web.Configuration.WebConfigurationManager.OpenWebConfiguration(confString); var section = (System.Web.Configuration.ScriptingJsonSerializationSection)conf.GetSection("system.web.extensions/scripting/webServices/jsonSerialization"); section.MaxJsonLength = int.MaxValue; conf.Save();
ooXei1sh

2

ทางเลือก ASP.NET MVC 5 แก้ไข:

(Mine คล้ายกับคำตอบของ MFC ข้างต้นโดยมีการเปลี่ยนแปลงเล็กน้อย)

ฉันยังไม่พร้อมที่จะเปลี่ยนเป็น Json.NET ในตอนนี้และในกรณีของฉันเกิดข้อผิดพลาดระหว่างการร้องขอ วิธีที่ดีที่สุดในสถานการณ์ของฉันคือการแก้ไขจริง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แต่ด้วยการเพิ่มค่า app.setig 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;
    }
}

1
ขอบคุณมันใช้งานได้ ... ขอบคุณมาก @Maxim Gershkovich
Jasper Manickaraj

1

โซลูชันสำหรับ WebForms UpdatePanel:

เพิ่มการตั้งค่า Web.config:

<configuration>
  <appSettings>
    <add key="aspnet:UpdatePanelMaxScriptLength" value="2147483647" />
  </appSettings>
</configuration>

https://support.microsoft.com/en-us/kb/981884

ScriptRegistrationManager ชั้นเรียนมีรหัสต่อไปนี้:

// Serialize the attributes to JSON and write them out
JavaScriptSerializer serializer = new JavaScriptSerializer();

// Dev10# 877767 - Allow configurable UpdatePanel script block length
// The default is JavaScriptSerializer.DefaultMaxJsonLength
if (AppSettings.UpdatePanelMaxScriptLength > 0) {
    serializer.MaxJsonLength = AppSettings.UpdatePanelMaxScriptLength;
}  

string attrText = serializer.Serialize(attrs);

1

เราไม่ต้องการเปลี่ยนแปลงฝั่งเซิร์ฟเวอร์ใด ๆ คุณสามารถแก้ไขได้เพียงแก้ไขโดยไฟล์ web.config สิ่งนี้ช่วยฉันได้ ลองใช้ดู

<appSettings>
 <add key="aspnet:MaxJsonDeserializerMembers" value="2147483647" />
<add key="aspnet:UpdatePanelMaxScriptLength" value="2147483647" />
</appSettings>  

and   

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



0

แก้ไขสำหรับ ASP.NET MVC: หากคุณต้องการแก้ไขเฉพาะการกระทำที่ทำให้เกิดปัญหาจากนั้นให้โค้ดดังนี้:

public JsonResult GetBigJson()
{
    var someBigObject = GetBigObject();
    return Json(someBigObject);
}

คุณสามารถเปลี่ยนเป็น:

public JsonResult GetBigJson()
{
    var someBigObject = GetBigObject();
    return new JsonResult()
    {
        Data = someBigObject,
        JsonRequestBehavior = JsonRequestBehavior.DenyGet,
        MaxJsonLength = int.MaxValue
    };
}

และฟังก์ชั่นควรเหมือนกันคุณสามารถคืน JSON ที่ใหญ่กว่าเป็นการตอบสนอง


คำอธิบายขึ้นอยู่กับซอร์สโค้ด ASP.NET MVC: คุณสามารถตรวจสอบController.Jsonวิธีการใดที่ทำในซอร์สโค้ด ASP.NET MVC

protected internal JsonResult Json(object data)
{
    return Json(data, null /* contentType */, null /* contentEncoding */, JsonRequestBehavior.DenyGet);
}

มันกำลังเรียกController.Jsonวิธีอื่น ๆ:

protected internal virtual JsonResult Json(object data, string contentType, Encoding contentEncoding, JsonRequestBehavior behavior)
{
    return new JsonResult
    {
        Data = data,
        ContentType = contentType,
        ContentEncoding = contentEncoding,
        JsonRequestBehavior = behavior
    };
}

ที่ผ่านcontentTypeและวัตถุcontentEncoding nullดังนั้นโดยทั่วไปเรียกในการควบคุมจะเทียบเท่ากับการโทรreturn Json(object) return new JsonResult { Data = object, JsonRequestBehavior = sonRequestBehavior.DenyGet }คุณสามารถใช้รูปแบบที่สองและพารามิเตอร์JsonResultparameterize

ดังนั้นจะเกิดอะไรขึ้นเมื่อคุณตั้งค่าMaxJsonLengthคุณสมบัติ (โดยค่าเริ่มต้นมันเป็นโมฆะ)? มันถูกส่งผ่านไปยังJavaScriptSerializer.MaxJsonLengthคุณสมบัติแล้วJavaScriptSerializer.Serializeวิธีการที่เรียกว่า :

JavaScriptSerializer serializer = new JavaScriptSerializer();
if (MaxJsonLength.HasValue)
{
    serializer.MaxJsonLength = MaxJsonLength.Value;
}

if (RecursionLimit.HasValue)
{
    serializer.RecursionLimit = RecursionLimit.Value;
}

response.Write(serializer.Serialize(Data));

และเมื่อคุณไม่ได้ตั้งค่าMaxJsonLenghtคุณสมบัติของ serializer มันจะใช้ ค่าเริ่มต้นซึ่งเป็นเพียง 2MB


-2

ถ้าค่า maxJsonLength นี้เป็น int ดังนั้นขนาดใหญ่ของมันคือ 32 บิต / 64 บิต / 16 บิต .... ฉันแค่ต้องการให้แน่ใจว่าค่าสูงสุดคืออะไรฉันสามารถตั้งค่าเป็น maxJsonLength ของฉัน

<scripting>
        <webServices>
            <jsonSerialization maxJsonLength="2147483647">
            </jsonSerialization>
        </webServices>
    </scripting>

-4

คุณไม่จำเป็นต้องทำอะไรกับ web.config คุณสามารถใช้คุณสมบัติสั้น ๆ ในระหว่างการจับค่าของรายการส่งผ่านได้ตัวอย่างเช่นประกาศรุ่นเช่น

public class BookModel
    {
        public decimal id { get; set; }  // 1 

        public string BN { get; set; } // 2 Book Name

        public string BC { get; set; } // 3 Bar Code Number

        public string BE { get; set; } // 4 Edition Name

        public string BAL { get; set; } // 5 Academic Level

        public string BCAT { get; set; } // 6 Category
}

ที่นี่ฉันใช้สัดส่วนสั้น ๆ เช่น BC = บาร์โค้ด BE = ฉบับหนังสือและอื่น ๆ


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