предисловие
Название не ошибся, очень хочется написатьbug
!
Когда я впервые получил эту просьбу, я не почувствовал ни малейшего волнения в своем сердце и даже был немного взволнован. Это моя специальность, наконец-то я могу писать открытоbug
🙄.
Давайте сначала посмотрим, что он собирается делать.На самом деле, основная цель состоит в том, чтобы заставить некоторые серверы с очень низкой нагрузкой потреблять дополнительные ресурсы, такие как память и процессор (насчет предыстории, я не буду говорить больше), чтобы можно было увеличить нагрузку.
Обзор распределения памяти JVM
Поэтому я взял шаттл и написал код, который примерно выглядит следующим образом:
После того, как я написал это, я думал о проблеме, коде вmem
Будет ли объект переработан сразу после выполнения метода? Я думаю, что должны быть некоторые люди, которые думают, что он перерабатывается после выполнения метода.
Я также пошел на серьезное расследование и спросил некоторых друзей; как и ожидалось, некоторые из них думают, что он перерабатывается после выполнения метода.
А факты? Я сделал тест.
Я только что запустил приложение со следующими параметрами запуска.
java -Djava.rmi.server.hostname=10.xx.xx.xx
-Djava.security.policy=jstatd.all.policy
-Dcom.sun.management.jmxremote.authenticate=false
-Dcom.sun.management.jmxremote.ssl=false
-Dcom.sun.management.jmxremote.port=8888
-Xms4g -Xmx4g -jar bug-0.0.1-SNAPSHOT.jar
Таким образом, я могу удаленно подключиться к этому приложению через порт JMX, чтобы наблюдать за ситуацией с памятью и GC.
Если метод завершен, он будет переработанmem
объект, когда я назначаю250M
памяти, память будет иметь четкую кривую, и GC тоже будет работать.
Затем наблюдайте за кривой памяти.
Выяснится, что действительно имеется значительное увеличение, но оно не восстанавливается сразу после этого, а поддерживается на этом уровне воды. При этом ГК слева не отвечает.
использоватьjstat
То же самое касается просмотра схемы памяти.
не важно какYGC,FGC
Нет, это всего лишь увеличение использования площади Эдема, ведь выделено 250М памяти.
Так как же он будет перерабатываться?
Я наблюдал кривую памяти после повторного выделения двух 250M.
Когда будут найдены третьи 250МEden
достигнутая площадь98.83%
Поэтому его необходимо переработать, когда он снова будет выделен.Eden
район производстваYGC
.
В то же время кривая памяти также была уменьшена.
Весь процесс преобразования показан на рисунке:
Поскольку инициализированная куча памяти4G
, поэтому рассчитаноEden
Площадь составляет примерно1092M
ОЗУ.
плюс запуск приложенияSpring
потребляет ок.20%
память, поэтому выделение 250M памяти 3 раза приведет кYGC
.
Вернемся к предыдущему вопросу:
mem
Поскольку объект не будет переработан после выполнения метода, когда он будет переработан?
На самом деле, помните только одно:Для объектов требуется сборщик мусораGC
могут быть переработаны, независимо от того, является ли объект локальной переменной или глобальной переменной.
В ходе предыдущих экспериментов также было установлено, что приEden
Недостаточно места в районеYGC
будут переработаны, когда мы создадимmem
объект.
Но на самом деле здесь есть скрытое условие: то есть объектлокальная переменная. Если объект является глобальной переменной, он все равно не может быть переработан.
Это то, что мы часто говоримОбъект недоступен, так что недостижимый объект находится вGC
Когда это происходит, считается, что это объект, который необходимо переработать, и перерабатывается.
После долгих размышлений, почему некоторые люди думают, что локальные переменные будут переработаны после выполнения метода?
Я думаю это надо путать, на самом деле после выполнения метода рекавери идет栈帧
.
Его самый непосредственный результат должен привести кmem
Этот объект больше не упоминается. Но отсутствие справки не означает, что она будет переработана сразу, то есть необходимость генерирования упомянутого вышеGC
будут переработаны.
Таким образом, используется вышеупомянутый недостижимый объектАлгоритмы анализа достижимостичтобы указать, какие объекты должны быть переработаны.
Когда на объект не ссылаются, он считается недостижимым.
Вот движущаяся картинка, которая более четкая:
После выполнения методаmem
Объект эквивалентен изображениюObject 5
, так что вGC
будет переработано вовремя.
Приоритет отдается размещению объектов в районе Эдема.
На самом деле из вышеприведенного примера видно, что в новом поколении объект предпочтительно размещается в районе Эдема, но есть посылка, что объект не может быть слишком большим.
Я уже писал об этом:
Крупные предметы уходят прямо в старость
Крупный объект напрямую отнесен к старому возрасту (насколько он велик, это можно настроить параметрами).
Когда я напрямую выделил 1000 МБ памяти, поскольку область Eden не могла быть загружена напрямую, вместо этого она была выделена в старом поколении.
можно увидетьEden
Площадь почти не изменилась, а вот старость увеличилась на 37%, согласно рассчитанной ранее памяти старости.2730M
Почти1000M
памяти.
проверка памяти линукс
Вернемся к тому, что мне нужно сделать на этот раз: увеличить нагрузку на сервер и ЦП.
С ЦП все в порядке, он сам использует определенное количество ресурсов, и каждый раз, когда создается объект, он потребляет некоторое количество ЦП.
В основном память, давайте сначала посмотрим на ситуацию с памятью перед запуском приложения.
Вероятно, используется только память 3G.
После запуска приложения оно, вероятно, потребляет всего около 600 м памяти.
Мне нужно выделить немного памяти, чтобы удовлетворить спрос, но здесь это немного сложно.
Невозможно выделять память все время, что приведет к слишком высокой загрузке ЦП, а память будет занята не очень сильно из-за рециркуляции сборщика мусора.
Поэтому мне нужно выделить небольшую сумму, чтобы большинство объектов в молодом поколении нужно было сохранить на 80% или 90%, чтобы их не собирали.
В то же время также необходимо выделить некоторые крупные объекты для старости и сохранить использование старости на уровне 80-90%.
Таким образом, память кучи 4G может использоваться в максимальной степени.
Поэтому я сделал следующее:
- Сначала выделите несколько небольших объектов в новом поколении (800M), чтобы поддерживать новое поколение на уровне 90%.
- а затем выделил
老年代内 *(100%-已使用的28%);也就是 2730*60%=1638M
Пусть старость тоже будет в районе 90%.
Эффект как выше.
Главное один разGC
Этого не произошло, и я этого хотел.
Конечное потребление памяти составляет около 3,5G.
Суммировать
Хотя спрос в этот раз довольно странный, но хочется точного контроляJVM
Распределение памяти по-прежнему не так просто.
Вам нужно иметь определенное представление о его структуре памяти и утилизации.Процесс написания этого бага действительно усилил впечатление.Если вам это поможет, пожалуйста, не скупитесь на лайки и репост.
Ваши лайки и репост - лучшая поддержка для меня