Java Virtual Machine - Optimieren des GC

Im letzten Kapitel haben wir verschiedene Generations-Gcs kennengelernt. In diesem Kapitel werden wir diskutieren, wie der GC eingestellt wird.

Wurfgröße

Die Größe des Heapspeichers ist ein wichtiger Faktor für die Leistung unserer Java-Anwendungen. Wenn es zu klein ist, wird es häufig gefüllt und muss daher häufig vom GC gesammelt werden. Wenn wir dagegen nur den Heap vergrößern, obwohl er weniger häufig gesammelt werden muss, würde sich die Länge der Pausen erhöhen.

Darüber hinaus hat das Erhöhen der Heap-Größe eine schwerwiegende Auswirkung auf das zugrunde liegende Betriebssystem. Durch die Verwendung von Paging lässt das Betriebssystem unsere Anwendungsprogramme viel mehr Speicher sehen, als tatsächlich verfügbar ist. Das Betriebssystem verwaltet dies, indem es Speicherplatz auf der Festplatte verwendet und inaktive Teile der Programme darauf kopiert. Wenn diese Teile benötigt werden, kopiert das Betriebssystem sie von der Festplatte zurück in den Speicher.

Nehmen wir an, eine Maschine verfügt über 8 GB Arbeitsspeicher, und die JVM sieht 16 GB virtuellen Speicher. Die JVM würde nicht wissen, dass tatsächlich nur 8 GB auf dem System verfügbar sind. Es wird nur 16G vom Betriebssystem anfordern, und sobald es diesen Speicher erhält, wird es ihn weiterhin verwenden. Das Betriebssystem muss viele Daten ein- und austauschen, was eine enorme Leistungsminderung für das System darstellt.

Und dann kommen die Pausen, die während der vollständigen GC eines solchen virtuellen Speichers auftreten würden. Da der GC auf den gesamten Heap zum Sammeln und Komprimieren wirkt, muss er viel warten, bis der virtuelle Speicher von der Festplatte ausgelagert wird. Bei einem gleichzeitigen Kollektor müssen die Hintergrundthreads viel warten, bis Daten aus dem Auslagerungsbereich in den Speicher kopiert wurden.

Hier stellt sich also die Frage, wie wir uns für die optimale Heap-Größe entscheiden sollen. Die erste Regel lautet, dem Betriebssystem niemals mehr Speicher anzufordern, als tatsächlich vorhanden ist. Dies würde das Problem des häufigen Austauschs vollständig verhindern. Wenn auf dem Computer mehrere JVMs installiert sind und ausgeführt werden, ist die Gesamtspeicheranforderung aller zusammen geringer als der tatsächlich im System vorhandene RAM.

Sie können die Größe der Speicheranforderung durch die JVM mithilfe von zwei Flags steuern.

  • -XmsN - Steuert den angeforderten Anfangsspeicher.

  • -XmxN - Steuert den maximalen Speicher, der angefordert werden kann.

Die Standardwerte dieser beiden Flags hängen vom zugrunde liegenden Betriebssystem ab. Für 64b-JVMs, die unter MacOS ausgeführt werden, ist -XmsN = 64M und -XmxN = mindestens 1 G oder 1/4 des gesamten physischen Speichers.

Beachten Sie, dass die JVM automatisch zwischen den beiden Werten wechseln kann. Wenn beispielsweise festgestellt wird, dass zu viel GC stattfindet, wird die Speichergröße weiter erhöht, solange sie unter -XmxN liegt und die gewünschten Leistungsziele erreicht werden.

Wenn Sie genau wissen, wie viel Speicher Ihre Anwendung benötigt, können Sie -XmsN = -XmxN festlegen. In diesem Fall muss die JVM keinen „optimalen“ Wert des Heaps ermitteln, und daher wird der GC-Prozess etwas effizienter.

Generationsgrößen

Sie können entscheiden, wie viel von dem Heap Sie dem YG zuweisen möchten und wie viel davon Sie dem OG zuweisen möchten. Diese beiden Werte wirken sich auf folgende Weise auf die Leistung unserer Anwendungen aus.

Wenn die Größe des YG sehr groß ist, wird es weniger häufig gesammelt. Dies würde dazu führen, dass weniger Objekte in die OG befördert werden. Wenn Sie dagegen die Größe von OG zu stark erhöhen, dauert das Sammeln und Verdichten zu lange und dies führt zu langen STW-Pausen. Somit muss der Benutzer ein Gleichgewicht zwischen diesen beiden Werten finden.

Unten finden Sie die Flags, mit denen Sie diese Werte festlegen können.

  • -XX:NewRatio=N: Verhältnis von YG zu OG (Standardwert = 2)

  • -XX:NewSize=N: YGs Anfangsgröße

  • -XX:MaxNewSize=N: YGs maximale Größe

  • -XmnN: Setzen Sie NewSize und MaxNewSize mit diesem Flag auf den gleichen Wert

Die anfängliche Größe des YG wird durch den Wert von NewRatio durch die gegebene Formel bestimmt -

(total heap size) / (newRatio + 1)

Da der Anfangswert von newRatio 2 ist, gibt die obige Formel an, dass der Anfangswert von YG 1/3 der gesamten Heap-Größe beträgt. Sie können diesen Wert jederzeit überschreiben, indem Sie die Größe des YG mithilfe des NewSize-Flags explizit angeben. Dieses Flag hat keinen Standardwert. Wenn es nicht explizit gesetzt ist, wird die Größe des YG weiterhin mit der obigen Formel berechnet.

Permagen und Metaspace

Das Permagen und der Metaspace sind Heap-Bereiche, in denen die JVM die Metadaten der Klassen speichert. Der Raum wird in Java 7 als "Permagen" und in Java 8 als "Metaspace" bezeichnet. Diese Informationen werden vom Compiler und der Laufzeit verwendet.

Sie können die Größe des Permagens mithilfe der folgenden Flags steuern: -XX: PermSize=N und -XX:MaxPermSize=N. Die Größe von Metaspace kann gesteuert werden mit:-XX:Metaspace- Size=N und -XX:MaxMetaspaceSize=N.

Es gibt einige Unterschiede, wie das Permagen und der Metaspace verwaltet werden, wenn die Flag-Werte nicht gesetzt sind. Standardmäßig haben beide eine Standardanfangsgröße. Während der Metaspace so viel vom Heap belegen kann, wie benötigt wird, kann das Permagen nicht mehr als die Standardanfangswerte einnehmen. Beispielsweise verfügt die 64b-JVM über 82 MB Heap-Speicherplatz als maximale Permagengröße.

Beachten Sie, dass ein Fehler aufgrund von Speichermangel auftreten kann, da der Metaspace unbegrenzt viel Speicher belegen kann, sofern nicht anders angegeben. Eine vollständige GC findet statt, wenn die Größe dieser Regionen geändert wird. Wenn während des Startvorgangs viele Klassen geladen werden, kann der Metaspace daher die Größe ändern, was jedes Mal zu einem vollständigen GC führt. Daher dauert es sehr lange, bis große Anwendungen gestartet sind, wenn die anfängliche Größe des Metaspaces zu niedrig ist. Es ist eine gute Idee, die anfängliche Größe zu erhöhen, da dies die Startzeit verkürzt.

Obwohl das Permagen und der Metaspace die Klassenmetadaten enthalten, sind sie nicht permanent, und der Speicherplatz wird vom GC wie bei Objekten zurückgefordert. Dies ist normalerweise bei Serveranwendungen der Fall. Wenn Sie eine neue Bereitstellung auf dem Server vornehmen, müssen die alten Metadaten bereinigt werden, da neue Klassenlader jetzt Speicherplatz benötigen. Dieser Platz wird vom GC freigegeben.