Generator mit virtuellem Thread

Barista

Top Contributor
Seltsamerweise findet man (fand ich) im Netz kaum Beispiele für Generatoren, die es in anderen Programmiersprachen gibt oder die man zum Beispiel in C# mit yield leicht realisieren kann.

Es geht darum, dass man mehrere Werte (Objekte) in einem Iterator zurück gibt, wobei der Iterator zwischen den next-Aufrufen den aktuellen Status vermerken und wieder aufnehmen muss. Man kann nicht einfach eine for-Schleife hinschreiben. Rekursiv wird es ganz übel (Iterator über Baum-Elemente).

Ich weiß nicht, ob es mit Streams eine bessere Lösung gibt.

Mit den neuen virtuellen Threads sollte dies einfach sein:

Java:
package kh.gen_virt_thrd;

import java.util.Iterator;
import java.util.NoSuchElementException;
import java.util.concurrent.LinkedBlockingDeque;

/**
 * Abstract class to generate values
 * with support of virtual thread.
 */
abstract public class GeneratorWthVrtThrd<T>
implements Iterable<T>
{
    /**
     * Method to implement to generate values.
     *
     * @param queue the output queue for the generated values
     */
    abstract public void generate(
            final GeneratorDeque<T> queue );

    @Override
    public final Iterator<T> iterator()
    {
        final GeneratorDeque<T> newQueue = new GeneratorDeque<T>();

        Thread.ofVirtual().start( () -> this.generate( newQueue ) );

        return new Iterator<T>()
        {
            private final GeneratorDeque<T> queue = newQueue;

            @Override
            public boolean hasNext()
            {
                return ! this.queue.isClosed();
            }

            @Override
            public T next()
            {
                if ( ! this.hasNext() )
                {
                    throw new NoSuchElementException();
                }

                return this.queue.take();
            }
        };
    }

    /**
     * Queue class to transfer generated values to consumer.
     */
    public static class GeneratorDeque<T>
    {
        private final LinkedBlockingDeque<T> innerQueue = new LinkedBlockingDeque<T>( 1 );

        private boolean isClosed;

        public void close()
        {
            this.isClosed = true;
        }

        public boolean isClosed()
        {
            return this.isClosed;
        }

        /**
         * Put value in queue.
         * Blocks if queue has no space for specified value.
         *
         * @param e value to put
         */
        public void put(
                final T e )
        {
            if ( this.isClosed )
            {
                throw new IllegalStateException( "queue already closed" );
            }

            try
            {
                this.innerQueue.put( e );
                // bei der Variante mit GeneratorDeque als top level Klasse war dieses sleep notwendig
                //Thread.sleep( 1 );
            }
            catch ( final InterruptedException exc )
            {
                throw new RuntimeException( exc );
            }
        }

        /**
         * Takes value from queue.
         * Blocks if queue has no value.
         *
         * @return value from queue
         */
        public T take()
        {
            if ( this.isClosed )
            {
                throw new IllegalStateException( "queue already closed" );
            }

            try
            {
                return this.innerQueue.take();
            }
            catch ( final InterruptedException exc )
            {
                throw new RuntimeException( exc );
            }
        }
    }

}

Hier der Code in Anwendung:

Java:
package kh.gen_virt_thrd;

/**
 * Spike for {@link GeneratorWthVrtThrd}.
 */
public class Integer0to2Generator
extends GeneratorWthVrtThrd<Integer>
{
    /**
     * @param args
     */
    public static void main(final String[] args)
    {
        // use output buffer to avoid blocking main thread for io operations
        final StringBuilder buff = new StringBuilder();

        for ( final Integer i : new Integer0to2Generator() )
        {
            //System.out.println( i );
            buff.append( i );
            buff.append( '\n' );
        }

        System.out.println( buff );
    }

    @Override
    public void generate(
            final GeneratorDeque<Integer> queue )
    {
        for ( int i = 0 ; i < 3 ; i++ )
        {
            //System.out.println( "queue.put: " + i );
            queue.put( i );
            //Thread.sleep( 1 );
        }
        //System.out.println( "queue.close" );
        queue.close();
    }

}

Erst hatte ich die static encapsulated Klasse GeneratorDeque als eigene top level Klasse.

Da hat der Code nur funktioniert, wenn ich das Thread.sleep( 1 ) aufgerufen habe (mit Thread.sleep( 0 ) ) hat es auch nicht funktioniert).

Der main-Thread hat blockiert, Es gab nur die Ausgaben 0, 1 und dann nix mehr, aber das Programm lief noch.

Komischerweise musste ich Thread.sleep( 1 ) nicht mehr aufrufen, als ich die Klasse GeneratorDeque zur eingeschlossenen Klasse gemacht habe.

Ich bin mir jetzt unsicher, ob das Konstrukt nicht doch irgendwann ungewollt blockiert.

Die jetzt auskommentierten System.out.println haben den Code auch funktionieren lassen, klar, bei IO blockiert der virtuelle Thread, gibt also die Kontrolle ab.

Wie kann ich es wirklich richtig machen?
 

Oneixee5

Top Contributor
Wenn ich das richtig sehe willst du erreichen, dass deine sequentielle Eingabe (put) - zwar in einem anderen Thread - trotzdem wieder sequentiell abgearbeitet wird. Das ganze in einem virtuellen Thread. Nun ist es so, das hier ein Thread eigentlich gar nicht notwendig wäre aber die willst einen non-blocking-input erreichen. Die Ausgabe soll dann im Main-Thread erfolgen.
Virtuelle Threads sind für die kurzzeitige Nutzung mit einer konkreten kompakten Aufgabe gedacht. Es werden direkt Vorteile der CPU genutzt und es entsteht minimaler Overhead. Um sich an diese Konzept zu halten, solltest du bei jedem put statt der Queue jedes Mal einen neuen virtuellen Thread erzeugen. Allerdings ist dann nicht garantiert, dass die Ausgabe in der Reihenfolge der Eingabe erfolgt.

Zur Funktion: LinkedBlockingDeque implementiert bereits Iterable, von daher ist GeneratorDeque eigentlich überflüssig. Für die Funktonalität queue.close(); sehe ich keine Notwendigkeit. Der Ausgabe-Thread ist hier der Main-Thread. Dieser wird also blockiert, wenn keine Eingaben in der Queue vorhanden sind. Das sehe ich problematisch, denn damit blockiert man diesen. Nur deshalb benötigst du close(); Wäre es nicht besser, wenn die Ausgabe in einem eigenem Thread ablaufen würde? Damit wäre der Main-Thread nie blockiert.

Ich denke die Struktur ist hier falsch gewählt. Es sollte natürlich eine BlockingQueue geben aber eingaben sollten von mind. einem Producer in der Queue abgelegt werden und dann von einem oder mehr Consumern der Queue wieder entnommen werden. Das Ende des Programms wird dadurch ermöglicht, dass auf allen noch laufenden Producer-Threads und Consumer-Threads interrupted aufgerufen wird. Dabei ist es praktisch, wenn man für die Erzeugung Threadgroup verwendet, so behält man die Kontrolle über alle eigene laufenden Threads.
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
Buroto Arrays generator Allgemeine Java-Themen 10
E Random Generator Allgemeine Java-Themen 6
L Generator für einen Parser implementieren Allgemeine Java-Themen 13
1 Name Generator für Videos Allgemeine Java-Themen 1
S QR-Code generator Allgemeine Java-Themen 2
A Audio Rechteck Generator Allgemeine Java-Themen 5
D Parser-generator für mathematische Funktionen Allgemeine Java-Themen 12
borobudur MVC Model Generator Allgemeine Java-Themen 2
boxi Registartions CodeBild generator Allgemeine Java-Themen 2
G Scanner-Generator zur Erkennung von Java Tokens Allgemeine Java-Themen 7
B Von neumann generator Allgemeine Java-Themen 7
F rxtx library mit virtuellem Comport Allgemeine Java-Themen 2
B Problem mit Virtuellem COM Port Allgemeine Java-Themen 1
R 11 GB File lesen ohne zu extrahieren Filedaten Bereich für Bereich adressieren dann mit Multi-Thread id die DB importieren Allgemeine Java-Themen 3
urmelausdemeis Exception in thread "main" java.lang.Error: Unresolved compilation problem: Allgemeine Java-Themen 7
smarterToby Wie stoppe ich diesen Thread Allgemeine Java-Themen 4
A Thread.sleep Problem Allgemeine Java-Themen 2
J Thread started nur einmal Allgemeine Java-Themen 19
W Server-Thread schreibt nicht alle Dateien Allgemeine Java-Themen 6
OnDemand Logfile pro User / Thread Allgemeine Java-Themen 7
OnDemand Thread / Service abbrechen Allgemeine Java-Themen 3
Thallius Ist meine static Helper Class Thread save? Allgemeine Java-Themen 9
P Swing Exception in thread "AWT-EventQueue-0" java.lang.IndexOutOfBoundsException: npoints > xpoints.length || npoints > ypoints.length Allgemeine Java-Themen 5
B Thread.sleep() in EJB Container wie lösen? Allgemeine Java-Themen 11
S Ist das Neuzuweisen von Feldern atomic und damit Thread-Safe? Allgemeine Java-Themen 2
S Exception in thread "main" java.lang.NullPointerException at FamilienApp.main(FamilienApp.java:15) Allgemeine Java-Themen 1
J Einen Thread in einer Schleife Allgemeine Java-Themen 2
E HILFE !! Exception in thread "main" java.lang.NoClassDefFoundError: org/apache/commons/io/FileUtils Allgemeine Java-Themen 4
Flynn Thread-Problem... Allgemeine Java-Themen 2
G Thread-Programmierung Allgemeine Java-Themen 5
S Datei wird nicht gefunden Thread.currentThread().getContextClassLoader().getResourceAsStream() Allgemeine Java-Themen 1
G Beendet sich der Thread selbst?! Allgemeine Java-Themen 3
mrbig2017 Sleep wird ignoriert und der Thread wartet nicht Allgemeine Java-Themen 1
S Thread beenden Allgemeine Java-Themen 9
M Array aus Thread Objekten erstellen Allgemeine Java-Themen 2
Aruetiise Swing JOptionPane ohne denn Thread zu pausieren Allgemeine Java-Themen 1
M Nanosekunden-Pause innerhalb einen Thread-Loops Allgemeine Java-Themen 3
E Thread Exception Allgemeine Java-Themen 6
javaerd Binomialkoeffizient ausrechnen, Exception in thread "main" java.lang.StackOverflowError Allgemeine Java-Themen 6
T Merkwürdiges Thread-Verhalten Allgemeine Java-Themen 6
K Thread Problem Allgemeine Java-Themen 6
W Thread sleep 30 sekunden - wenn keine Antwort bis dahin neu senden Allgemeine Java-Themen 2
H Thread bleibt stehen bei jar in jar Allgemeine Java-Themen 1
J Threads HTTP Request (Thread) dauert lange - in Android Allgemeine Java-Themen 3
F CPU Last eines Thread ausfindig machen Allgemeine Java-Themen 0
V Compiler-Fehler Exception in thread "AWT-EventQueue-0" java.lang.IndexOutOfBoundsException: Index: 125, Size: 125 Allgemeine Java-Themen 11
Tausendsassa Threads Einen Thread sich selbst schließen lassen Allgemeine Java-Themen 17
P Threads BufferedImage, Thread Concurrency Allgemeine Java-Themen 1
M Klasse in separaten Thread ausführen.Wie genau? Allgemeine Java-Themen 2
llabusch Thread blockiert Dialog Allgemeine Java-Themen 1
J Thread wait() Allgemeine Java-Themen 2
V Thread.sleep und InterruptedException? Allgemeine Java-Themen 1
G Thread nicht von GC zerstört Allgemeine Java-Themen 6
J Wie erschaffe ich einen sicheren Datenaustausch zwischen Thread und Nicht-Threads Allgemeine Java-Themen 8
Sogomn Thread blocken bis Taste gedrückt Allgemeine Java-Themen 5
T Starten vom Thread Allgemeine Java-Themen 3
T Wait/Notify() bei Thread Allgemeine Java-Themen 6
J Exception in thread "main" java.lang.NoClassDefFoundError Allgemeine Java-Themen 4
M Exception in thread "AWT-EventQueue-0" Allgemeine Java-Themen 6
Q Thread wacht nicht auf Allgemeine Java-Themen 7
T Fragen zum Thread-Thema Allgemeine Java-Themen 4
T Threads Input/Output im Thread - Datei ohne Inhalt Allgemeine Java-Themen 1
T Fragen zum Thread-Thema Allgemeine Java-Themen 9
C Threads Variablen in einem Thread Aktualisieren Allgemeine Java-Themen 17
U Thread beenden Allgemeine Java-Themen 3
W Threads Mit Thread und Runtime externe Programme öffnen Allgemeine Java-Themen 0
N Thread interrupt Status debuggen Allgemeine Java-Themen 6
A Thread: Code paralell ausführen in mehreren Instanzen Allgemeine Java-Themen 1
E Threads linkedlist/multi-thread problem Allgemeine Java-Themen 3
B Erkennen, wann Prozess beendet ist, dann Thread beenden. Allgemeine Java-Themen 6
A Thread Fehler absichtlich provozieren Allgemeine Java-Themen 3
B Threads Java Thread kommunizieren Allgemeine Java-Themen 12
N Thread Sicherheit im komplexen Datenmodell Allgemeine Java-Themen 7
K Thread richtig benutzen Allgemeine Java-Themen 3
K Exception in thread "AWT-EventQueue-1" Allgemeine Java-Themen 2
vandread Problem bei kleiner Thread-Übung Allgemeine Java-Themen 2
G Thread erzeugt nicht plausible NullPointerException Allgemeine Java-Themen 7
H Netbeans Warning bei Thread.sleep in Schleife Allgemeine Java-Themen 4
P [Thread] Scheint nicht Sequenziell zu Arbeiten Allgemeine Java-Themen 9
A eine test thread.join() frage Allgemeine Java-Themen 2
tuttle64 Verständnisprobleme mit Thread Locks Allgemeine Java-Themen 4
G Threads Thread bei Datenabfrage Allgemeine Java-Themen 3
S Thread anhalten per Button ? Allgemeine Java-Themen 3
E Thread Programmierung Allgemeine Java-Themen 2
S Threads ServerSocket-Thread soll schlafen, bis er gebraucht wird Allgemeine Java-Themen 2
V Thread schneller stoppen Allgemeine Java-Themen 2
V anstatt thread.join() einfach while schleife? Allgemeine Java-Themen 8
B Mausbewegung im Thread erkennen (hoch/runter) Allgemeine Java-Themen 6
G Linux/C++/Pthreads auf JVM zugreifen, thread safe? Allgemeine Java-Themen 10
K Threads Probleme mit Thread Allgemeine Java-Themen 13
K Threads Thread überprüfen Allgemeine Java-Themen 3
Z Threads Thread für einen Client Allgemeine Java-Themen 9
M Thread JavaFish Allgemeine Java-Themen 10
G Thread.sleep Allgemeine Java-Themen 12
M Threads Viele Aufrufe aus Thread, komisches Verhalten Allgemeine Java-Themen 8
B Threads Main Thread warten auf abgebrochen Task warten lassen Allgemeine Java-Themen 25
K Timer Thread Allgemeine Java-Themen 8
M Methoden Static Methoden und Thread??? Allgemeine Java-Themen 4
N java.lang.IllegalMonitorStateException: object not locked by thread before notify() Allgemeine Java-Themen 2
C Mehothode in anderenm Thread aufrufen Allgemeine Java-Themen 10

Ähnliche Java Themen


Oben