GUI Änderungen und Threads

Jens81

Gesperrter Benutzer
Hallo zusammen,

ich habe eine Frage zu GUI Änderungen und Threads: Ich weiß mittlerweile, dass ich keine GUI Änderungen innerhalb von Threads durchführen darf, bzw. wenn ich dies tue muss ich sie mit invokeLater / invokeAndWait aufrufen (oder gleich einen SwingWorker verwenden).
Meine (vielleicht ja einfach zu beantwortende) Frage: Was genau sind denn (relevante) Gui Änderungen? Das Ändern eines Datensatzes im TableModel? Das Setzen eines Strings in ein JTextField? Das Setzen eines Hakens einer JCheckBox? Wie wird das abgegrenzt?

Gruß,
Jens
 
G

Gastredner

Gast
Ich denke, jede Änderung, die eine Änderung in der Anzeige der GUI nach sich zieht, ist eine relevante Änderung.
 

Michael...

Top Contributor
Das ist nicht so einfach zu beantworten.

z.B. ist die Methode setText() bei allen Komponenten die von JTextComponent erben Thread safe, setSelected() an einer JComboBox oder setText() an einem JLabel scheinbar nicht.

Auch bei Änderungen am TableModel z.B. per table.setValueAt(x,y) ist ein invokeLater o.ä. nicht notwendig, da das Neuzeichnen der Tabelle im EDT abläuft. Welcher Mechanismus dahinter steckt, k.A.
 

Jens81

Gesperrter Benutzer
Gilt dies nur für setValueAt?
Das Ändern im TableModel mit removeRow z.B. erfordert ein invokeLater

Wie kann ich denn rausfinden, wann ich invokeLater benötige? In der Api steht zumindest bei den Methoden nichts dazu.
 
B

bygones

Gast
Hallo zusammen,

ich habe eine Frage zu GUI Änderungen und Threads: Ich weiß mittlerweile, dass ich keine GUI Änderungen innerhalb von Threads durchführen darf, bzw. wenn ich dies tue muss ich sie mit invokeLater / invokeAndWait aufrufen (oder gleich einen SwingWorker verwenden).
Meine (vielleicht ja einfach zu beantwortende) Frage: Was genau sind denn (relevante) Gui Änderungen? Das Ändern eines Datensatzes im TableModel? Das Setzen eines Strings in ein JTextField? Das Setzen eines Hakens einer JCheckBox? Wie wird das abgegrenzt?

Gruß,
Jens
hae ? alles ist in Threads - natuerlich kannst/musst du GUI Aenderungen in Threads abwickeln.

Ich denke was du meinst ist, dass man groessere Berechnungen bzw zeitintensive Arbeiten nicht im GUI Thread laufen lassen darf, denn dann erhaelt man das schone "Gui ist eingefroren" Problem.

Wenn du ein einfaches setText('hallo') oder ein setSelected() aufrufst musst du dich um nix kuemmern.

Wenn du eine 1GB datei einlesen willst, dies in einem Progressbar anzeigen lassen willst, dann musst du dich um das richtige Threadmanagement kuemmern
 

Michael...

Top Contributor
Eigentlich sollte man sicherstellen, dass non thread safe Swingmethoden im EDT aufgerufen werden. Allerdings ist es so, dass solche Methoden wie setValue, addRow, removeRow (um mal beim Bsp. JTable zu bleiben) zwar nicht thread safe sind, dahinter aber ein Listenerkonzept steckt. D.h. Änderungen am Model führen nicht zu direkten Änderungen an der GUI, sondern lösen Events aus, welche die GUI ändern und Events laufen im EDT und somit auch die Änderung an der GUI.

Und daher muss ich bygones zustimmen: problematisch ist eher, dass rechenintensive Aktivitäten den EDT blockieren und man diese daher besser in separate Threads auslagert.
Und ob jetzt ein setText oder setSelected im EDT läuft oder nicht, darüber würde ich mir auch keine allzu grossen Gedanken machen.

Zur Info: um herauszufinden ob der Code gerade im EDT läuft oder nicht, gibt's
Java:
javax.swing.SwingUtilities.isEventDispatchThread()

Und hier noch ein Bsp. wie man sieht laufen die Methoden addRow, removeRow und setValueAt zwar nicht im EDT, dass renderen der Zelleninhalte wird dann allerdings im EDT ausgeführt:
Java:
import java.awt.Component;

import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.DefaultTableModel;

public class EDTTest extends JFrame {
	private DefaultTableModel model;
	
	public EDTTest () {
		model = new DefaultTableModel(new Object[][] {{"Zelle1"}, {"Zelle2"}},new String[] {"Test"}) {
			public void removeRow(int row) {
				super.removeRow(row);
				System.out.println("rem: " + javax.swing.SwingUtilities.isEventDispatchThread());
			}
			
			public void addRow(Object[] object) {
				super.addRow(object);
				System.out.println("add: " + javax.swing.SwingUtilities.isEventDispatchThread());
			}
			
			public void setValueAt(Object object, int row, int column) {
				super.setValueAt(object, row, column);
				System.out.println("set: " + javax.swing.SwingUtilities.isEventDispatchThread());
			}
		};
		JTable table;
		this.getContentPane().add(new JScrollPane(table = new JTable(model)));
		table.setDefaultRenderer(Object.class, new DefaultTableCellRenderer(){
			public Component getTableCellRendererComponent(JTable table, Object value,
                    boolean isSelected, boolean hasFocus, int row, int column) {
				System.out.println("rendering: " + javax.swing.SwingUtilities.isEventDispatchThread());
				return super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
			}
		});
		new Thread(new Runnable(){
			public void run() {
				while(true) {
					try {
						Thread.sleep(1000);
						model.addRow(new Object[] {"Inserted"});
						Thread.sleep(1000);
						model.setValueAt("new Value", model.getRowCount()-1, 0);
						Thread.sleep(1000);
						model.removeRow(model.getRowCount()-1);
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
			}
		}).start();
	}
	
	public static void main(String[] args) {
		JFrame frame = new EDTTest();
		frame.setBounds(0, 0, 500, 300);
		frame.setLocationRelativeTo(null);
		frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
		frame.setVisible(true);
	}
}
 

Jens81

Gesperrter Benutzer
Genau darum geht es ja. Es ist schon klar das alles in Threads läuft :)

Wenn ich nun einen zeitintensive Berechnung in einem eigenen Thread laufen lasse und während der Berechnung z.B. Zwischenergebnisse in eine TextBox schreiben möchte, oder eine Tabellenzeile ändern / löschen muss, sollte man ja wissen ob das nun in diesem Thread geschehen darf, oder ich in den EDT wechseln muss.

Das man TableModel Änderungen doch im EDt laufen lassen muss, habe ich aus diesem Thread hier: http://www.java-forum.org/awt-swing-swt/79700-jtable-auto-rowsorter-nullpointer-exception.html Dort gab es Probleme mit dem AutoTableRowSorter, wenn man die Änderungen nicht im EDT ausführt.
 

Marco13

Top Contributor
Eigentlich sollte man sicherstellen, dass non thread safe Swingmethoden im EDT aufgerufen werden. Allerdings ist es so, dass solche Methoden wie setValue, addRow, removeRow (um mal beim Bsp. JTable zu bleiben) zwar nicht thread safe sind, dahinter aber ein Listenerkonzept steckt. D.h. Änderungen am Model führen nicht zu direkten Änderungen an der GUI, sondern lösen Events aus, welche die GUI ändern und Events laufen im EDT und somit auch die Änderung an der GUI.
...
Und ob jetzt ein setText oder setSelected im EDT läuft oder nicht, darüber würde ich mir auch keine allzu grossen Gedanken machen.


Das stimmt so nicht. Es werden zwar Events gefeuert, aber die werden auf dem Thread gefeuert, der auch die Änderung macht. Und DESwegen sollte man die Änderungen auch nur auf dem EDT machen. Und darum sollte man sich schon Gedanken machen. Es gibt nicht viele allgemeine "Regeln", die man bei Swing beachten muss. Aber eine (und wichtige) ist eben die "Single Thread Rule":

Once a Swing component has been realized, all code that might affect or depend on the state of that component should be executed in the event-dispatching thread. ( Threads and Swing )

EDIT: Genaugenommen ist die sogar etwas unpräzise (oder zumindest mißverständlich). Entgegen der eigentlichen Aussage dieser Regel muss oft z.B. auch schon das Erstellen der Swing-Components auf dem EDT passieren.
 

Michael...

Top Contributor
Die Events werden zwar im laufenden Thread abgefeuert. Aber abgearbeitet werden sie doch im EDT? So verstehe ich zumindest folgenden Satz aus Deinem Verweis:
Most post-initialization GUI work naturally occurs in the event-dispatching thread. Once the GUI is visible, most programs are driven by events such as button actions or mouse clicks, which are always handled in the event-dispatching thread.
 

Marco13

Top Contributor
Ja, der entscheidende Teil ist eben das
"...most programs are driven by events such as button actions or mouse clicks, which are always handled in the event-dispatching thread.

Wenn man in der "actionPerformed" bei einem Buttonklick sowas macht wie
[c]someTable.insertRow(n, someData);[/c]
dann passiert das schon auf dem EDT, und die Swing-Welt ist in Ordnung.

Wenn man aber in einem eigenen Thread sowas macht wie
[c]someTable.insertRow(n, someData);[/c]
dann ist der Ablauf relativ leicht nachvollziehbar:

Java:
class DefaultTableModel extends AbstractTableModel
{
    public void insertRow(int row, Vector rowData) {
        ...
        fireTableRowsInserted(row, row);
    }
}

class AbstractTableModel ...
{
    public void fireTableRowsInserted(int firstRow, int lastRow) {
        fireTableChanged(...);
    }

    public void fireTableChanged(TableModelEvent e) {
        ....
	for (int i = listeners.length-2; i>=0; i-=2) {
	    if (listeners[i]==TableModelListener.class) {
		((TableModelListener)listeners[i+1]).tableChanged(e);
	    }
	}
    }
}

Auf dem Thread, auf dem das
[c]someTable.insertRow(n, someData);[/c]
gemacht wird, wird auch das "tableChanged" vom TableModelListener (also von der JTable) aufgerufen. Das eigentliche Neuzeichnen der Table wird dann zwar durch ein repaint() ausgelöst und findet damit auf dem EDT statt, aber die anderen Änderungen an der JTable passieren eben auf diesem "falschen" Thread - wenn ein anderer Thread zufällig im gleichen Moment eine andere Änderung macht, kann's halt krachen (und tut es oft auch - sobald die Software beim Kunden laufen soll :D )
 

Jens81

Gesperrter Benutzer
Falls man eine Tabelle via Click auf den ColumnHeader per DefaultRowSorter sortiert und danach eine TableModel Änderung im falschen Thread durchführt, kracht es nicht nur hin und wieder...

Vielen Dank an alle :)
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
J (Geplante) Änderungen an einer Datei vorübergehend speichern und anwenden? Allgemeine Java-Themen 12
C .Ini-Datei Wert Änderungen werden nicht angenommen Allgemeine Java-Themen 3
R Änderungen in einem Verzeichnis beobachten - Windows Allgemeine Java-Themen 3
P Variablen in einer anderen Klasse auf Änderungen überwachen Allgemeine Java-Themen 12
martin82 Java Runtime Update >17 - SwingWorker Änderungen? Allgemeine Java-Themen 7
A java.io-Änderungen zwischen java 1.4 und 1.6 Allgemeine Java-Themen 18
D Änderungen an einer lokalen Datei abprüfen/überwachen Allgemeine Java-Themen 4
N Applet übernimmt keine Änderungen Allgemeine Java-Themen 13
G TreeSet ändert sich bei Änderungen nicht! Allgemeine Java-Themen 15
S Änderungen im Source-Code direkt verwenden können? Allgemeine Java-Themen 3
rode45e Java Threads Allgemeine Java-Themen 4
M Threads Allgemeine Java-Themen 1
L Threads Threads in Chatroom Allgemeine Java-Themen 30
berserkerdq2 run-methode eines Threads so programmieren, dass 30x die Sekunde etwas ausgeführt wird. Allgemeine Java-Themen 44
berserkerdq2 Threads, wie genau läuft das in Java ab? (Ich kann Threads erstellen und nutzen, nur das Verständnis) Allgemeine Java-Themen 6
CptK Backpropagation parallelisieren: Kommunikation zwischen den Threads Allgemeine Java-Themen 7
J Eine Frage zu den Threads und Task Allgemeine Java-Themen 1
W Wieviele Threads sind sinnvoll? Allgemeine Java-Themen 8
W Alternative für Threads Allgemeine Java-Themen 6
V Threads Probleme beim Aufrufen von Methoden einer anderen Klasse (Threads) Allgemeine Java-Themen 14
T Multithreading: Wie viele Threads sollte ich erstellen? Allgemeine Java-Themen 12
G Threads vom Mainprogramm steuern Allgemeine Java-Themen 8
S BlockingQueue mit dynamischer Anpassung der Anzahl von Producer und Consumer Threads Allgemeine Java-Themen 1
x46 Threads Threads anhalten Allgemeine Java-Themen 1
J Threads verbessern die Performance NICHT ? Allgemeine Java-Themen 8
W Threads Problem Allgemeine Java-Themen 15
T Threads Tic Tac Toe mit Threads Allgemeine Java-Themen 1
M Threads über Kommandozeile Allgemeine Java-Themen 5
mrbig2017 Threads Chat Programm mit Threads? Allgemeine Java-Themen 2
J Threads - java.lang.IllegalThreadStateException Allgemeine Java-Themen 6
J Internet Broswer in Threads öffnen Allgemeine Java-Themen 1
B Threads Multithreading Threads sollen warten Allgemeine Java-Themen 12
N 1000 MQTT Messages die Sekunde - 1000 Threads erstellen ? Allgemeine Java-Themen 10
D Threads Parallel laufende Threads Allgemeine Java-Themen 4
J Unvorhersehbares Verhalten - benutze ich die falsche Bedingungsprüfung oder brauche ich Threads? Allgemeine Java-Themen 12
D Eine Forschleife mit Threads abarbeiten um es zu schneller zu machen. Ist das möglich? Allgemeine Java-Themen 20
S Wie kann ich eine kleine Stelle in meinem Code mit multiplen Threads abarbeiten..? Allgemeine Java-Themen 20
P Threads Parallelisierte DB-Abfragen mit variabler Anzahl an Threads Allgemeine Java-Themen 4
J Threads Threads Allgemeine Java-Themen 9
Viktim Threads Liste In unterschiedlichen Threads bearbeiten Allgemeine Java-Themen 23
E Threads Ausführung in Threads ist langsamer als ohne Threads Allgemeine Java-Themen 13
A Anzahl an Threads Systemweit Allgemeine Java-Themen 2
Tausendsassa Input/Output Problem mit der gleichzeitigen Ausgabe zweier Threads Allgemeine Java-Themen 8
S Alle Methodenaufrufe eines Threads notieren..? Allgemeine Java-Themen 7
M Threads JPanel eingeforen mit Threads Allgemeine Java-Themen 2
F Threads Allgemeine Java-Themen 6
F Threads Allgemeine Java-Themen 2
M Sinn von Threads? Allgemeine Java-Themen 1
J Wie erschaffe ich einen sicheren Datenaustausch zwischen Thread und Nicht-Threads Allgemeine Java-Themen 8
L Abfragen ob Threads fertig Allgemeine Java-Themen 3
P Threads Java Zugreifen Allgemeine Java-Themen 6
K Problem: Java-Klasse mit mehreren Threads als eigenen Prozess starten Allgemeine Java-Themen 3
K KeyEvent in Threads Allgemeine Java-Themen 11
V Threads Weshalb funktionieren meine Threads nicht? Allgemeine Java-Themen 2
Thallius Speicherverhalten von Properties und mehreren Threads Allgemeine Java-Themen 5
L Threads beenden Allgemeine Java-Themen 4
P Threads Threads nicht gleichzeitig starten Allgemeine Java-Themen 3
S Threads Threads werden nicht beendet Allgemeine Java-Themen 2
S Start des zweiten Threads erst nach Beenden des ersten Threads Allgemeine Java-Themen 13
N Threads statische Methoden in Threads Allgemeine Java-Themen 5
P 4 Threads in einer Methode Allgemeine Java-Themen 2
M Eclipse Mehrere Threads, mehrere Konsolen Allgemeine Java-Themen 4
OnDemand Threads und synchronized Allgemeine Java-Themen 9
R LinkedList und Threads: Strukturprobleme bez. löschen von Elementen Allgemeine Java-Themen 3
R LinkedList und Threads - welche Methode ist besser? Allgemeine Java-Themen 2
OnDemand Threads und synvhronized Allgemeine Java-Themen 2
S Problem mit Threads Allgemeine Java-Themen 1
W Threads Threads warten lassen Allgemeine Java-Themen 5
H Optimierung durch Threads Allgemeine Java-Themen 31
B Threads halten sich irgendwie auf... Allgemeine Java-Themen 6
M Threads Allgemeine Java-Themen 8
K JNI: Methoden aus unterschiedlichen Threads aufrufen Allgemeine Java-Themen 3
A Applet Alle Threads beim schließen des Applets beenden Allgemeine Java-Themen 8
A Problem mit der Synchronisierung von Threads Allgemeine Java-Themen 15
R SecurityManager für einzelne Klassen/Threads? Allgemeine Java-Themen 38
O Threads und If Befehle Allgemeine Java-Themen 7
P Threads abwechseln lassen mit wait() und notify() Allgemeine Java-Themen 2
H Sehr viele Threads effizient Verwalten Allgemeine Java-Themen 13
C Threads und Exceptions Allgemeine Java-Themen 7
H java.lang.OutOfMemoryError bei der wiederholten Erzeugng von Threads Allgemeine Java-Themen 8
S Threads Abarbeitungsstatus von Threads in Datei schreiben Allgemeine Java-Themen 2
M Zugriff zweier Threads auf diesselbe Methode Allgemeine Java-Themen 16
E Threads Sudoku Threads Allgemeine Java-Themen 8
M Java Threads - Wait Notify - Verständnisproblem Allgemeine Java-Themen 5
Gossi Threads mit unterschiedlichen Aufgaben in einer Klasse? Allgemeine Java-Themen 9
G Threads Ablauf von Threads im Spezialfall Allgemeine Java-Themen 4
V Threads bei quadcore Allgemeine Java-Themen 10
V 1000 Threads oder Iterativ? Allgemeine Java-Themen 11
4 Simple(?) Frage zu Threads Allgemeine Java-Themen 14
B Threads Game of Life - Threads Allgemeine Java-Themen 49
R Threads Exceptions von Threads abfangen im ThreadPool Allgemeine Java-Themen 5
S Threads Ende sämtlicher Threads abwarten Allgemeine Java-Themen 6
S Frage zu Threads (Sichtbarkeit und Verhalten) Allgemeine Java-Themen 11
M Java-Threads und Datentypen-Zugriffe Allgemeine Java-Themen 7
P Threads- Programming Allgemeine Java-Themen 2
G Threads Klasse Sound und Threads bleiben hängen Allgemeine Java-Themen 4
C Threads Zwei Threads greifen auf LinkedList zu. Allgemeine Java-Themen 12
M OutOfMemoryError in nebenläufigen Threads Allgemeine Java-Themen 6
M Threads dauerhafte bewegung mit threads Allgemeine Java-Themen 11
frankred Threads Auf eine Gruppe von Threads warten Allgemeine Java-Themen 11

Ähnliche Java Themen

Neue Themen


Oben