Анатомия "утечки памяти"
В перспективе.NET:
- Что такое утечка памяти?
- Как вы можете определить, протекает ли ваше приложение? Каковы эффекты?
- Как вы можете предотвратить утечку памяти?
- Если у вашего приложения есть утечка памяти, оно исчезнет, когда процесс завершится или будет убит? Или утечки памяти в вашем приложении влияют на другие процессы в системе даже после завершения процесса?
- А как насчет неуправляемого кода, доступ к которому осуществляется через COM Interop и / или P/Invoke?
15 ответов
Лучшее объяснение, которое я видел, находится в главе 7 бесплатной электронной книги "Основы программирования".
По сути, в .NET происходит утечка памяти, когда ссылочные объекты укоренены и, следовательно, не могут быть удалены сборщиком мусора. Это происходит случайно, когда вы держите ссылки за пределами предполагаемой области.
Вы будете знать, что у вас есть утечки, когда вы начнете получать исключения OutOfMemoryException или использование памяти превысит ваши ожидания (PerfMon имеет хорошие счетчики памяти).
Понимание модели памяти .NET- ваш лучший способ избежать этого. В частности, понимание того, как работает сборщик мусора и как работают ссылки, - опять же, я отсылаю вас к главе 7 электронной книги. Кроме того, помните об общих подводных камнях, вероятно, наиболее распространенных из них. Если объект A зарегистрирован в событии на объекте B, то объект A будет задерживаться, пока объект B не исчезнет, потому что B содержит ссылку на A. Решение состоит в том, чтобы отменить регистрацию ваших событий, когда вы закончите.
Конечно, хороший профиль памяти позволит вам увидеть ваши графы объектов и изучить вложение / ссылки на ваши объекты, чтобы увидеть, откуда приходят ссылки и какой корневой объект ответственен ( профиль муравьев красных ворот, JetBrains dotMemory, http://memprofiler.com/ действительно хороши выбор, или вы можете использовать только текстовые WinDbg и SOS, но я настоятельно рекомендую коммерческий / визуальный продукт, если вы не настоящий гуру).
Я считаю, что неуправляемый код подвержен типичным утечкам памяти, за исключением того, что общие ссылки управляются сборщиком мусора. Я могу ошибаться в этом последнем пункте.
Строго говоря, утечка памяти потребляет память, которая "больше не используется" программой.
"Больше не используется" имеет более одного значения, это может означать "больше нет ссылки на него", то есть полностью не подлежит восстановлению, или это может означать "ссылка, восстановление", неиспользование, но программа все равно сохраняет ссылки. Только последнее применимо к.Net для идеально управляемых объектов. Однако не все классы идеальны, и в какой-то момент базовая неуправляемая реализация может привести к постоянной утечке ресурсов для этого процесса.
Во всех случаях приложение потребляет больше памяти, чем строго необходимо. Побочные эффекты, в зависимости от утечки количества, могут переходить от нуля к замедлению, вызванному чрезмерным сбором, к серии исключений памяти и, наконец, к фатальной ошибке, сопровождаемой принудительным завершением процесса.
Вы знаете, что у приложения есть проблема с памятью, когда мониторинг показывает, что все больше памяти выделяется вашему процессу после каждого цикла сборки мусора. В этом случае вы либо слишком много храните в памяти, либо какая-то неуправляемая реализация не работает.
Для большинства утечек ресурсы восстанавливаются, когда процесс завершается, однако некоторые ресурсы не всегда восстанавливаются в некоторых точных случаях, дескрипторы курсора GDI печально известны для этого. Конечно, если у вас есть механизм межпроцессного взаимодействия, память, выделенная в другом процессе, не будет освобождена, пока этот процесс не освободит его или не прекратит работу.
Я думаю, что на вопросы "что такое утечка памяти" и "каковы последствия" уже хорошо ответили, но я хотел бы добавить еще несколько вещей по другим вопросам...
Как понять, протекает ли ваше приложение
Один интересный способ - открыть perfmon и добавить трассировки для # байтов во всех кучах и коллекциях # Gen 2, в каждом случае рассматривая только ваш процесс. Если использование определенной функции приводит к увеличению общего количества байтов, и эта память остается выделенной после следующей коллекции 2-го поколения, вы можете сказать, что эта функция теряет память.
Как предотвратить
Другие хорошие мнения были даны. Я бы просто добавил, что, пожалуй, наиболее часто пропускаемая причина утечек памяти.NET - это добавление обработчиков событий к объектам без их удаления. Обработчик событий, прикрепленный к объекту, является формой ссылки на этот объект, поэтому предотвращает сбор даже после того, как все другие ссылки исчезли. Всегда не забывайте отключать обработчики событий (используя -=
синтаксис в C#).
Утечка исчезает при выходе из процесса, а как насчет взаимодействия COM?
Когда ваш процесс завершается, вся память, отображаемая в его адресное пространство, восстанавливается ОС, включая любые COM-объекты, обслуживаемые из DLL. Сравнительно редко COM-объекты могут обслуживаться из отдельных процессов. В этом случае, когда ваш процесс завершается, вы все равно можете нести ответственность за память, выделенную в любых процессах COM-сервера, которые вы использовали.
Я бы определил утечки памяти как объект, не освобождающий всю память, выделенную после ее завершения. Я обнаружил, что это может произойти в вашем приложении, если вы используете Windows API и COM (т. Е. Неуправляемый код, в котором есть ошибка или неправильно управляется), в среде и в сторонних компонентах. Я также обнаружил, что проблема может быть в том, чтобы не убирать после использования определенных предметов, таких как ручки.
Я лично перенес исключения из памяти, которые могут быть вызваны, но не исключают утечки памяти в приложениях dot net. (OOM также может прийти от пиннинга, см. Pinning Artical). Если вы не получаете ошибок OOM или вам необходимо подтвердить, является ли это утечкой памяти, вызывающей ее, тогда единственный способ - профилировать ваше приложение.
Я также попытался бы обеспечить следующее:
a) Все, что реализует Idisposable, удаляется либо с помощью блока finally, либо с помощью оператора using, в том числе кистей, ручек и т. д. (некоторые люди утверждают, что вообще ничего не устанавливают)
б) Все, что имеет метод close, снова закрывается с помощью оператора finally или оператора using (хотя я обнаружил, что использование метода не всегда закрывается в зависимости от того, объявлен ли объект вне оператора using)
c) Если вы используете неуправляемый код / Windows API, то после этого они будут корректно обработаны. (у некоторых есть методы очистки для освобождения ресурсов)
Надеюсь это поможет.
Если вам нужно диагностировать утечку памяти в.NET, проверьте эти ссылки:
https://web.archive.org/web/20141203155344/http://msdn.microsoft.com/en-us/magazine/cc163833.aspx
https://web.archive.org/web/20150102222036/http://msdn.microsoft.com/en-us/magazine/cc164138.aspx
В этих статьях описывается, как создать дамп памяти вашего процесса и как его проанализировать, чтобы вы могли сначала определить, является ли утечка неуправляемой или управляемой, и, если она управляемой, как выяснить, откуда она поступает.
Microsoft также имеет новый инструмент для создания аварийных дампов, заменяющий ADPlus, который называется DebugDiag.
Использование CLR Profiler от Microsoft http://www.microsoft.com/downloads/details.aspx?familyid=86ce6052-d7f4-4aeb-9b7a-94635beebdda&displaylang=en - отличный способ определить, какие объекты содержат память, к чему приводит поток выполнения на создание этих объектов, а также мониторинг, какие объекты живут где-то в куче (фрагментация, LOH и т. д.).
Наилучшим объяснением того, как работает сборщик мусора, является Jeff Richters CLR через C# book (гл. 20). Чтение этого дает большую основу для понимания того, как объекты сохраняются.
Одной из наиболее распространенных причин случайного укоренения объектов является соединение событий вне класса. Если вы подключите внешнее событие
например
SomeExternalClass.Changed += new EventHandler(HandleIt);
и не забудьте отцепить его, когда вы утилизируете, тогда SomeExternalClass имеет ссылку на ваш класс.
Как упоминалось выше, профилировщик памяти SciTech отлично показывает корни объектов, которые, как вы подозреваете, протекают.
Но есть также очень быстрый способ проверить конкретный тип - просто использовать WnDBG (вы можете даже использовать это в окне немедленного доступа VS.NET, когда подключено):
.loadby sos mscorwks
!dumpheap -stat -type <TypeName>
Теперь сделайте то, что, по вашему мнению, удалит объекты этого типа (например, закройте окно). Здесь удобно иметь кнопку отладки, которая будет запускаться System.GC.Collect()
Пару раз.
Тогда беги !dumpheap -stat -type <TypeName>
снова. Если число не уменьшилось или не снизилось так сильно, как вы ожидаете, у вас есть основания для дальнейшего расследования. (Я получил этот совет с семинара, проведенного Инго Раммером).
Я полагаю, что в управляемой среде утечка может привести к тому, что вы сохраните ненужную ссылку на большой кусок памяти.
Почему люди думают, что утечка памяти в.NET отличается от любой другой утечки?
Утечка памяти - это когда вы подключаетесь к ресурсу и не отпускаете его. Вы можете сделать это как в управляемом, так и в неуправляемом кодировании.
Что касается.NET и других инструментов программирования, были идеи о сборке мусора и других способах минимизации ситуаций, которые могут привести к утечке вашего приложения. Но лучший метод предотвращения утечек памяти - это то, что вам нужно понимать основную модель памяти и то, как все это работает, на платформе, которую вы используете.
Вера в то, что GC и другая магия очистит ваш бардак, - это короткий путь к утечкам памяти, который будет трудно найти позже.
При кодировании неуправляемого вы обычно убираетесь, вы знаете, что ресурсы, за которые вы беретесь, будут вашими обязанностями, а не уборщиком.
В.NET, с другой стороны, многие думают, что GC очистит все. Ну, это немного для вас, но вы должны убедиться, что это так. .NET делает много вещей, поэтому вы не всегда знаете, имеете ли вы дело с управляемым или неуправляемым ресурсом, и вам необходимо убедиться, с чем вы имеете дело. Обращение со шрифтами, ресурсами GDI, активным каталогом, базами данных и т. Д. Обычно является тем, на что нужно обратить внимание.
В управляемых терминах я поставлю свою шею на линии, чтобы сказать, что она исчезнет, как только процесс будет убит / удален.
Я вижу, что у многих есть это, и я действительно надеюсь, что это закончится. Вы не можете попросить пользователя закрыть ваше приложение, чтобы навести порядок! Посмотрите на браузер, который может быть IE, FF и т. Д., Затем откройте, скажем, Google Reader, оставьте его на несколько дней и посмотрите, что произойдет.
Если затем вы откроете другую вкладку в браузере, зайдете на какой-нибудь сайт, а затем закроете вкладку, на которой размещена другая страница, из-за которой произошла утечка браузера, думаете ли вы, что браузер освободит память? Не так с IE. На моем компьютере IE легко съест 1 ГБ памяти за короткий промежуток времени (около 3-4 дней), если я использую Google Reader. Некоторые газеты еще хуже.
Я согласен с Бернардом в том, что в.net будет утечка памяти.
Вы могли бы профилировать свое приложение, чтобы видеть использование его памяти, и определить, что если оно управляет большим количеством памяти, когда это не должно быть, вы могли бы сказать, что оно имеет утечку.
В управляемых терминах я поставлю свою шею на линии, чтобы сказать, что она исчезнет, как только процесс будет убит / удален.
Неуправляемый код - это его собственный зверь, и если в нем есть утечка, он будет следовать стандартному мем. определение утечки.
Я полагаю, что в управляемой среде утечка может привести к тому, что вы сохраните ненужную ссылку на большой кусок памяти.
Абсолютно. Кроме того, неиспользование метода.Dispose() для одноразовых объектов, когда это необходимо, может вызвать утечку памяти. Самый простой способ сделать это с помощью блока using, потому что он автоматически выполняет.Dispose() в конце:
StreamReader sr;
using(sr = new StreamReader("somefile.txt"))
{
//do some stuff
}
И если вы создаете класс, который использует неуправляемые объекты, если вы не реализуете IDisposable правильно, вы можете вызывать утечки памяти для пользователей вашего класса.
Все утечки памяти устраняются завершением программы.
Утечка достаточной памяти, и операционная система может решить проблему от вашего имени.
Также имейте в виду, что.NET имеет две кучи, одна из которых является кучей больших объектов. Я полагаю, что в эту кучу помещаются объекты размером примерно 85 КБ или больше. Эта куча имеет другие правила жизни, чем обычная куча.
Если вы создаете большие структуры памяти (словарь или список), было бы разумно поискать точные правила.
Что касается освобождения памяти при завершении процесса, если только вы не используете Win98 или ее эквивалент, все возвращается к ОС после завершения. Единственными исключениями являются вещи, которые открыты кросс-процессами, а у другого процесса ресурс остается открытым.
COM-объекты могут быть сложными, хотя. Если вы всегда используете IDispose
шаблон, вы будете в безопасности. Но я наткнулся на несколько сборок взаимодействия, которые реализуют IDispose
, Ключ здесь, чтобы позвонить Marshal.ReleaseCOMObject
когда вы закончите с этим. COM-объекты по-прежнему используют стандартный подсчет ссылок COM.
Я обнаружил, что .Net Memory Profiler очень помогает при обнаружении утечек памяти в.Net. Он не бесплатный, как Microsoft CLR Profiler, но, на мой взгляд, более быстрый и точный.
Одно из определений: невозможно освободить недоступную память, которая больше не может быть выделена новому процессу во время выполнения процесса выделения. В основном это можно вылечить с помощью методов ГХ или обнаружить с помощью автоматизированных инструментов.
Для получения дополнительной информации, пожалуйста, посетите http://all-about-java-and-weblogic-server.blogspot.in/2014/01/what-is-memory-leak-in-java.html.