Heim >Java >javaLernprogramm >Warum werden meine stark erreichbaren Java 8-Objekte vorzeitig finalisiert?

Warum werden meine stark erreichbaren Java 8-Objekte vorzeitig finalisiert?

Mary-Kate Olsen
Mary-Kate OlsenOriginal
2024-12-12 13:45:161021Durchsuche

Why Are My Strongly Reachable Java 8 Objects Being Finalized Prematurely?

„Finalize() wird in Java 8 für stark erreichbare Objekte aufgerufen“

Problem:

Bei einer mit Java 7 entwickelten und kürzlich auf Java 8 aktualisierten Anwendung treten gelegentlich Ausnahmen auf, die darauf hinweisen, dass ein Stream geschlossen wurde vorzeitig. Untersuchungen haben ergeben, dass der Finalizer-Thread finalize() vorzeitig für ein Objekt aufruft, das den Stream enthält, was den Abschluss auslöst.

Hintergrund:

Die Codestruktur beinhaltet einen MIME Writer (MIMEWriter), ein MIME-Körperteil (MIMEBodyPart), und einen Eingabestream (InflaterInputStream), der eine angehängte Datei darstellt. MIMEBodyPart erweitert HTTPMessage, das eine close()-Methode enthält, die den zugrunde liegenden Stream schließt. Darüber hinaus verfügt HTTPMessage über eine finalize()-Methode, die versucht, close() für den Stream aufzurufen, wenn er noch geöffnet ist.

Abfolge von Ereignissen:

  1. MIMEWriter schreibt Header für den Anhangsteil.
  2. MIMEBodyPart schreibt den Hauptinhalt mit IOUtil.copy, das Blöcke aus dem Eingabestream in die Ausgabe kopiert stream.
  3. IOUtil.copy versucht, einen Block zu lesen, stößt jedoch auf den geschlossenen Stream, was die Ausnahme auslöst.

Ursache:

Die Die Methode MIMEBodyPart.finalize() wird vom Finalizer-Thread vorzeitig aufgerufen, während IOUtil.copy aktiv ausgeführt wird. Java 8 führte Garbage-Collection-Optimierungen ein, die es ermöglichen, dass Objekte finalisiert werden, auch wenn sie noch durch lokale Variablen oder aktive Methodenaufrufe referenziert werden.

Das MIMEBodyPart-Objekt ist tatsächlich über den Stapelrahmen von MIMEBodyPart.writeBodyPartContent erreichbar, was impliziert, dass das JVM sollte nicht versuchen, es fertigzustellen. Da der Verweis auf MIMEBodyPart innerhalb der IOUtil.copy-Schleife jedoch nicht aktiv verwendet wird, ist er nicht mehr erreichbar und für die Garbage Collection und Finalisierung geeignet.

Folgen:

Die Eine vorzeitige Finalisierung kann zu fehlerhaftem Verhalten und fehlerhaften Daten führen Verlust.

Lösung:

Der empfohlene Ansatz besteht darin, die selbst erstellte Bibliothek erneut zu besuchen und die Verwendung von finalize()-Methoden zu eliminieren. Da die MIME-Bibliothek von Java Mail das Problem nicht aufwies, könnte sie als Alternative dienen.

Alternative Vermutung:

Eine weitere mögliche Erklärung betrifft den InflaterInputStream. Wenn die MIMEBodyPart.finalize()-Methode während eines nicht unterbrechbaren Vorgangs in InflaterInputStream aufgerufen wird, könnte sie den Stream unterbrechen und die Ausnahme auslösen. Diese Hypothese bedarf jedoch weiterer Untersuchungen.

Das obige ist der detaillierte Inhalt vonWarum werden meine stark erreichbaren Java 8-Objekte vorzeitig finalisiert?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn