Serializing an object using Json.Net causes Out of Memory exception

24,389

Solution 1

It is due to very large number of records you are trying to serialize, which occupies large memory. Solutions which I have found for this error as directly writing to the documents using StreamWriter(JsonWriter or TextWriter).

If you have Object use TextWrite

using (TextWriter textWriter = File.CreateText("LocalJsonFile.json"))
{
    var serializer = new JsonSerializer();
    serializer.Serialize(textWriter , yourObject);
}

If you have string use StringWriter

  StringBuilder sb = new StringBuilder();
  StringWriter sw = new StringWriter(sb);

  using(JsonWriter textWriter = new JsonTextWriter(sw))
  {
     var serializer = new JsonSerializer();
     serializer.Serialize(textWriter, yourObject);
  }

Solution 2

Updated Code based on suggestions in the comments on the question, This works!

public static async Task<bool> SerializeIntoJson<T>(string fileName, StorageFolder destinationFolder, Content content)
    {
        try
        {
            StorageFile file = await destinationFolder.CreateFileAsync(fileName, CreationCollisionOption.ReplaceExisting);
            using (var stream = await file.OpenStreamForWriteAsync())
            {

                StreamWriter writer = new StreamWriter(stream);
                JsonTextWriter jsonWriter = new JsonTextWriter(writer);
                JsonSerializer ser = new JsonSerializer();

                ser.Formatting = Newtonsoft.Json.Formatting.Indented;
                ser.PreserveReferencesHandling = PreserveReferencesHandling.Objects;
                ser.TypeNameHandling = TypeNameHandling.All;
                ser.Error += ReportJsonErrors;

                ser.Serialize(jsonWriter, content);

                jsonWriter.Flush();

            }
            return true;
        }
        catch (NullReferenceException nullException)
        {

            logger.LogError("Exception happened while serializing input object, Error: " + nullException.Message);
            return false;
        }
        catch (Exception e)
        {

            logger.LogError("Exception happened while serializing input object, Error: " + e.Message, e.ToString());
            return false;
        }
    }
Share:
24,389
Jack_2060
Author by

Jack_2060

Updated on November 04, 2020

Comments

  • Jack_2060
    Jack_2060 over 3 years

    Disclaimer: I did went through most of the solution provided here but most of them were talking about OOM exception while Deserialization.

    I am trying to serialize an object( it's a Tree) into Json using Json.Net. Everything works fine for small objects but i get OOM exception when i try it with large objects. As it works with smaller object of same datatype i am assuming there is no circular reference (I did inspect my data structure for it). Is there a way where i can convert my object into stream ( this is a Windows Store app ) and generate the Json using that stream ?

     public static async Task<bool> SerializeIntoJson<T>(string fileName, StorageFolder destinationFolder, Content content)
        {
            ITraceWriter traceWriter = new MemoryTraceWriter();
            try
            {
    
                string jsonString = JsonConvert.SerializeObject(content, Formatting.Indented, new JsonSerializerSettings
                {
                    PreserveReferencesHandling = PreserveReferencesHandling.Objects,
                    TypeNameHandling = TypeNameHandling.All,
                    Error = ReportJsonErrors,
                    TraceWriter = traceWriter,
                    StringEscapeHandling = StringEscapeHandling.EscapeNonAscii
                });
                System.Diagnostics.Debug.WriteLine(traceWriter);
    
                StorageFile file = await destinationFolder.CreateFileAsync(fileName, CreationCollisionOption.ReplaceExisting);
                await Windows.Storage.FileIO.WriteTextAsync(file, jsonString);
                return true;
            }
            catch (NullReferenceException nullException)
            {
                System.Diagnostics.Debug.WriteLine(traceWriter);
                logger.LogError("Exception happened while serializing input object, Error: " + nullException.Message);
                return false;
            }
            catch (Exception e)
            {
                System.Diagnostics.Debug.WriteLine(traceWriter);
                logger.LogError("Exception happened while serializing input object, Error: " + e.Message, e.ToString());
                return false;
            }
        }
    

    In order to convert my object into stream, the code i found out was using a BinaryFormatter which is not available in Windows store app dll's.