„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:
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!