Vala: создание потока приводит к потере свойств

В настоящее время я пытаюсь выучить некоторые вала. После прочтения руководства я захотел перенести некоторые приложения, написанные на java, на vala.

Поэтому я попытался написать приложение, которое отслеживает изменения в папке. Этот монитор должен создаваться как отдельный поток, чтобы приложение могло выполнять другие действия, связанные с мониторингом. Легкая работа в java.

После некоторого исследования того, как отслеживать (кстати, попал в stackoverflow), я получил следующий код:

Бегун:

public class App : Object 
    {
    private const string FOLDER_TO_WATCH = "/home/chr/Work/vala-monitor-test/tmp/";

    public App() {

        GLib.MainLoop loop = new GLib.MainLoop();

        stdout.printf("Folder: %s\n", FOLDER_TO_WATCH);

        try {
            var daemon = new MonitorDaemon( File.new_for_path( FOLDER_TO_WATCH ) );
            unowned Thread<void*> t = Thread.create<void*>( daemon.watch, true );
        }
        catch (ThreadError err) {
            stderr.printf("Error occured: %s", err.message);
        }
        catch (MonitorError err) {
            stderr.printf("Error occured: %s", err.message);
        }

        stdout.printf("waiting..\n");

        loop.run();

    }

    static int main( string[] args ) 
    {           

        if (!Thread.supported ()) {
            stderr.printf ("Cannot run without thread support.\n");
            return 1;
        }

        new App();
        return 0;

    }

}

Класс монитора:

errordomain MonitorError {
    NOT_A_DIRECTORY
}

public class MonitorDaemon : Object {

    // properties
    public File watched_folder { get; construct set; }

    // internal
    private bool _stop = false;


    public MonitorDaemon ( File folder_to_watch ) throws MonitorError {

        if ( folder_to_watch.query_file_type ( FileQueryInfoFlags.NONE ) != FileType.DIRECTORY ) {
            stderr.printf("Unable to watch: Not a directory\n");
            throw new MonitorError.NOT_A_DIRECTORY("Not a directory");
        }

        this.watched_folder = folder_to_watch;

        stdout.printf("Ready to watch %s\n", this.watched_folder.get_path() );

    }

    public void* watch() {

        try {
            FileMonitor monitor = this.watched_folder.monitor_directory ( FileMonitorFlags.NONE );
            monitor.changed.connect( event_happened );
        }
        catch (IOError err) {
            stderr.printf("Error occured: %s", err.message);
        }

        stdout.printf( "Started watching folder: %s\n", this.watched_folder.get_path() );

        while (!this._stop) { }

        stdout.printf( "Stopped watching folder: %s\n", this.watched_folder.get_path() );

        return null;

    }

    public void stop() {
        // cleanup..            
        this._stop = true;
    }           

    private void event_happened ( File file, File? other_file, FileMonitorEvent event_type ) {
        stdout.printf( "File '%s': %s\n", file.get_path(), event_type.to_string() );
    }

}

При создании потока для метода MonitorDaemon.watch свойство, установленное в конструкторе, снова имеет значение null, что приводит к следующему выводу в командной строке:

Folder: /home/chr/Work/vala-monitor-test/tmp/
Ready to watch /home/chr/Work/vala-monitor-test/tmp/
waiting..

(process:7159): GLib-GIO-CRITICAL **: g_file_monitor_directory: assertion `G_IS_FILE (file)' failed

(process:7159): GLib-GObject-WARNING **: invalid (NULL) pointer instance

(process:7159): GLib-GObject-CRITICAL **: g_signal_connect_object: assertion `G_TYPE_CHECK_INSTANCE (instance)' failed

(process:7159): GLib-GIO-CRITICAL **: g_file_get_path: assertion `G_IS_FILE (file)' failed
Started watching folder: (null)

person cr0    schedule 28.04.2012    source источник
comment
Решение. Не инициализируйте класс, содержащий fn, который будет выполняться как поток в том же предложении try..catch. Это вызывает вышеуказанные ошибки. Но почему???   -  person cr0    schedule 28.04.2012
comment
Вала считается ссылкой. При выходе из блока try{} он становится unref'd, а затем освобождается.   -  person lethalman    schedule 28.04.2012
comment
@lethalman: вы можете скопировать свой комментарий в ответ, чтобы его можно было принять.   -  person jacknad    schedule 30.04.2012
comment
Я не был уверен, что правильно понял вопрос.   -  person lethalman    schedule 01.05.2012


Ответы (1)


Вала считается ссылкой. При выходе из блока try{} он становится unref'd, а затем освобождается. Вы должны где-то держать MonitorDaemon живым.

person lethalman    schedule 30.04.2012