interesting OutOfMemoryException with StringBuilder

后端 未结 3 1696
时光取名叫无心
时光取名叫无心 2020-11-30 14:02

I have the need to continuously build large strings in a loop and save them to database which currently occasionally yields an OutOfMemoryException.

What

相关标签:
3条回答
  • 2020-11-30 14:23

    There is memory but no contiguous segment that can handle the size of your string builder. You have to know that each time the buffer of the string builder is too short, its size is doubled. If you can define (in the ctor) the size of your builder, it's better. You MAY call GC.Collect() when you are done with a large collection of objects.

    Actually, when you have an OutOfMemory, it generaly shows a bad design, you may use the hard drive (temp files) instead of memory, you shouldn't allocate memory again and again (try to reuse objects/buffers/...).

    I STRONGLY advice you to read this post “Out Of Memory” Does Not Refer to Physical Memory from Eric Lippert.

    0 讨论(0)
  • 2020-11-30 14:38

    Try to reuse StringBuilder object when you do data generation.

    After or before use just reset the size of the StringBuilder to 0 and start appending. This will decrease number of allocations and possibly make OutOfMemory situation very rare.

    To illustrate my point:

    void MainProgram()
    {
        StringBuilder builder = new StringBuilder(2 * 1024); //2 Kb
    
        PerformOperation(builder);
        PerformOperation(builder);
        PerformOperation(builder);
        PerformOperation(builder);
    }
    
    void PerformOperation(StringBuilder builder)
    {
        builder.Length = 0;
    
        //
        // do the work here builder.Append(...);
        //
    }
    
    0 讨论(0)
  • 2020-11-30 14:44

    With the sizes you mention you are probably running into Large Object Heap (LOH) fragmentation.

    Reusing StringBuilder objects is not a direct solution, you need to get a grip on the underlying buffers.
    If possible, calculate or estimate the size beforehand and pre-allocate.

    And it could help if you round up allocations, let's say to multiples of 20k or so. That could improve reuse.

    0 讨论(0)
提交回复
热议问题