Warum wird finalize() nicht aufgerufen?



  • Hallo,
    hat jemand eine Erklärung dafür, warum obj.finalize() in dem unterem Beispiel nicht aufgerufen wird?
    Gibt es einen andere Möglichkeit, auf das löschen eines Objekts zu reagieren?

    Danke

    (j2sdk1.4.2_03)

    public class Test {
    	public static void main(String[] args) {
    		MyObject obj = new MyObject();
    	}
    }
    
    public class MyObject {
    	protected void finalize() throws Throwable {
    		System.out.println("MyObject.finalize");
    		super.finalize();
    	}
    }
    


  • http://www.javaworld.com/javaworld/jw-06-1998/jw-06-techniques_p.html

    One more rule of thumb about finalizers concerns objects left on the heap at the end of the application's lifetime. By default, the garbage collector will not execute the finalizers of any objects left on the heap when the application exits. To change this default, you must invoke the runFinalizersOnExit() method of class Runtime or System, passing true as the single parameter. If your program contains objects whose finalizers must absolutely be invoked before the program exits, be sure to invoke runFinalizersOnExit() somewhere in your program.



  • Da diese Methode deprecated ist, und der GC in der default-Einstellungen finalize() vor Beenden der VM nicht aufruft, bleibt einem wohl nichts anderes
    übrig, als obj.finalize() (evtl umbenennen) "von Hand" aufzurufen. Was meiner Meinung nach kein besonders guter Stil ist, da man sich so auf den Benutzer der Klasse verlassen muss, und diese Besonderheit in der Dokumentation der Klasse besonders hervorheben muss.

    Sehe ich das richtig?

    http://java.sun.com/j2se/1.4.2/docs/api/java/lang/System.html#runFinalizersOnExit(boolean)

    public static void runFinalizersOnExit(boolean value)

    Deprecated. This method is inherently unsafe. It may result in finalizers being called on live objects while other threads are concurrently manipulating those objects, resulting in erratic behavior or deadlock.



  • Hi
    Das ist so ein kleineres Problem vom Java. das man keine Kontrolle hat wan ein Objekt genau gelöscht wird, und wann desen finalizer aufgerufen wird. fals überhautp.

    um Code beim beenden der Laufzeitumgebung auszufüren gibts es unter runtime sogenente ShutdownHooks. die man einbinden kann. sie sollen dazu dienen ggf. offengebleibende Netzwerkverbindungen, Dateien, Datenbankanbindungen ,... sicher zu beenden.

    Guss Termite


  • Mod

    Es gibt IMHO überhaupt keinen Grund, finalize zu nutzen. Kannst du mal sagen, wofür du das brauchst?



  • Frage: Wie sieht das eigentlich mit Destruktoren aus?? Wenn ich nen static counter hab, wieviel Objekte ich mit Konstruktoren erstelle, kann ich mich darauf verlassen, dass der Destruktor brav aufgerufen wird und meinen Zähler verringert, wenn ich alle Referenzen auf das Objekt lösche?? (wahrscheinlich nicht)



  • Optimizer: Es gibt in Java keine Destruktoren wie in C++. Es gibt nicht das Konzept, dass irgendwas passiert, wenn eine Variable aus dem Scope fällt oder eine Referenz umgebogen wird, wie bei C++-Smartpointern. Alle Referenzen sind dumme Pointer, und den Rest macht der GC, wenn er sich dazu bequemen kann. Dabei ruft er zwar Finalizer auf (so vorhanden), aber das ist, wie schon gesagt wurde, am Programmende nicht garantiert.



  • @Optimizer:
    In Java gibt es eben keine Destructoren.

    finalize() ist eine Methode die jedes Object hat. Diese Funktion wird vom Garbage-Collector aufgerufen, bevor er das Object löscht.
    Dieser löscht jedoch die Objecte nicht sofort, wenn sie nicht mehr referenziert werden, sondern "irgendwann". (Wenn er halt dazu kommt)

    Wenn man finalize() überschreibt kann man es etwa als Destruktor ansehen, wobei eben nicht garantiert ist, zu welchem Zeitpunkt er aufgerufen wird.

    @Gregor
    Eben um zu vermeiden, dass er Benutzer der Klasse ein explizietes obj.close()
    machen muss.
    In meinem Fall, sollte die Klasse etwas in eine Datei schreiben, was sie im Kontruktor dann wieder ausliest.

    @Terminte
    ShutdownHooks hab ich mir auch schon angeschaut, jedoch kann mir nicht vorstellen, wie man in der eigenen Klasse einen Shutdownhook auf den eigenen
    "Destruktor" setzen soll.

    Lange Rede kurzer Sinn:
    finalize() ist ziemlich nutzlos, da der Aufruf nicht garantiert ist,
    und man muss eine .close()-Methode anbieten.
    Java ist eben nicht c++ 😉



  • @Tendor: ... und das ist auch gut so.

    SCNR



  • Da widerspreche ich bestimmt nicht. Ich bin nur etwas erstaunt, wie man jetzt z.B. bequem einen Objektzähler programmieren soll.



  • Was soll man denn mit einem Objektzähler?


  • Mod

    BTW: Ohne es zu wissen, vermute ich fast, dass so ein "Objekte zählen" ab Java 1.5 sehr einfach wird. Ab da kommt die neue Management-API. Ich vermute, dass die soetwas generell kann.



  • hi

    um einen shutdown hook einzubinden must du einen Thread inplementieren. das geht wenn ich mich recht erinnere einmal über der implementieren des Interfaces runabel oder das ableiten von Thread

    beim interface ist das recht einfach. du implementiertst die metode run() wenn micht nicht alles teuscht. Die wird ausgefürt wenn der shutdown ausgelöst wird. da run() ja im gleichen obj implementiert wird wie der rest deines obj hast du volle zugriff drauf. und kannst somit die datei schliessen und den rest der daten speichern. Im konstruktor meldest du das objekt an, im finaliser trägst du in wieder aus. zu beachten ist, das die verwaltung etwas komplexer wird. "ist die datei offen dann eintragen des shutdown hooks(bevor die datei geöffnet wird)." "wird die datei geschlossen, alles sichern, datei schliesen und dann shutownhook abmelden." sicherzutsellen ist dann auch, das nichts passiert wenn die datei 2 mal geschlossen wird. bzw ist dies zuverhindern.

    wenn du von Thread ableitest, must du dem thread erst dine klasse bekantmachen.
    Deine klasse stellt entsprechende schnittstellen zur verfügung damit der Thread das speichern der daten einleiten kann. verhalten dan so änlich wie oben.

    Aufpassen dadlocks und schleiffen können die VM blokieren.

    gruss michael



  • Meinst du so:

    public class MyObject implements Runnable {
    	MyObject() {
    		Runtime.getRuntime().addShutdownHook(new Thread(this));
    	}
    
    	public void run() {
    		this.close();
    	}
    
    	private void close() {
    		System.out.println("MyObject.close");
    	}
    }
    

    Hast recht, das geht, aber ich weiß nicht... Ich denke das ist nicht besonders toller Stil. Dann doch lieber public void close(), was dann eben der Nutzer aufrufen muss.

    (Vielleicht geht es ja in Java 1.5 wieder ohne Tricksereien, hab mir die neuen Features noch nicht so genau angesehen)



  • Hi

    Ja so in etwa hab ich mir das vorgestellt.

    gruss Michael


Anmelden zum Antworten