Thread-Schreibe-Lese-Problem

Adriano10

Bekanntes Mitglied
Java:
class KontoSL {

    private int kontostand;

    private int summeBarEinzahlungen;

    private int summeUeberweisungsEinzahlungen;

    KontoSL() {
        this.kontostand = 0;
        this.summeBarEinzahlungen = 0;
        this.summeUeberweisungsEinzahlungen = 0;
    }

    void barEinzahlen(int betrag) {
        this.summeBarEinzahlungen = this.summeBarEinzahlungen + betrag;
        // Problem: Thread-Wechsel hier
        this.kontostand = this.kontostand + betrag;
    }

    int getProzentualeBarEinzahlungen() {
        if (this.kontostand == 0) {
            return 0;
        }
        return 100 * this.summeBarEinzahlungen / this.kontostand;
    }

    void perUeberweisung(int betrag) {
        this.summeUeberweisungsEinzahlungen = this.summeUeberweisungsEinzahlungen + betrag;
        this.kontostand = this.kontostand + betrag;
    }

    int getKontostand() {
        return this.kontostand;
    }

}

class Konto implements Runnable{
    KontoSL sl;
    private int betrag;

    public Konto(KontoSL sl, int betrag) {
        this.sl = sl;
        this.betrag = betrag;
    }

    @Override
    public void run() {
        int kontostand = 0;
        kontostand += betrag;
        sl.barEinzahlen(kontostand);
    }

    public static void main(String[] args) throws InterruptedException {
        KontoSL sl = new KontoSL();

        Thread thread = new Thread(new Konto(sl, 15));
        Thread thread1 = new Thread(new Konto(sl, 25));

        thread.start();
        thread1.start();
        thread.join();
        thread1.join();

        System.out.println(sl.getKontostand());
    }
}

Moin,

kann jemand mich bitte aufklären, warum meine Code funktioniert?
Result ist immer 40, was korrekt ist.

Normalerweise ohne Semaphore oder Synchronisation sollte richtiges Ergebnis nicht rauskommen. Was mache ich hier falsch?
 

fhoffmann

Top Contributor
Du machst nichts falsch; der Fehler tritt nur in ganz seltenen Fällen auf:
this.kontostand = this.kontostand + betrag;
Nur wenn im einen Thread this.kontostand gelesen wir, dann zum anderen Thread gewechselt wird, dort this.kontostand geändert wird, und danach wieder vom ersten Thread this.kontostand geschrieben wird, kann ein Fehler auftreten.
In deinem Fall wird das kaum passieren. Bei einer Bank, die tausende von Kontobewegungen am Tag hat, kann dies allerdings hin und wieder auftreten - und wäre sehr peinlich.
 

KonradN

Super-Moderator
Mitarbeiter
Das Ergebnis kann falsch sein. Man kann nicht vorher sagen, wann ein Thread Wechsel statt findet und wann nicht.

Bei Dir ist es so, dass du extrem wenig machst. Der erste Thread ist schon fertig, ehe der zweite wirklich gestartet werden kann. Und der Zeitpunkt, bei dem ein Threadwechsels zu Problemen führen würde, ist sehr klein.

Um da etwas zu forcieren, könntest Du einfach in dem Thread mehrere Einzahlungen machen oder einfach die Zeitspanne sehr groß machen, in der das Problem auftritt:
Java:
    void barEinzahlen(int betrag) {
        this.summeBarEinzahlungen = this.summeBarEinzahlungen + betrag;
        try {
            Thread.sleep(1000);
        } catch (Exception e) {}
        this.kontostand = this.kontostand + betrag;
    }
 

LimDul

Top Contributor
Das macht übrigens Threading-Probleme so unglaublich schwer zu analysieren und zu finden. Aus folgenden Gründen

Sie treten sehr selten auf
Das heißt sie schaffen es selten in direkt in den Fokus. Ein Fehler der nur sporadisch auftritt wird gerne mal mit "naja, lässt sich nicht reproduzieren" geschlossen. Da braucht es schon engagierte Entwickler und technisch versierte Test-Manager / Projektleitung die versteht das ein Fehler, den man nicht versteht potentiell eine tickende Zeitbombe sein kann.

Sie sind meist nicht reproduzierbar
Einen Fehler, den man nicht reproduzieren kann, kann man extrem schwer analysieren. Das einzige, was man machen kann, ist sich den Code ansehen und überlegen warum er sowas getan hat. Und der ist leider selten so gekapselt und simpel wie hier

Die Ursache ist oft ganz woanders/nicht in der Methode sichtbar
Meist schaut man sich den Code an und stellt fest, dass der eigentlich korrekt ist. Die Ursache ist eine parallele Ausführung - die aber oft technisch beliebig weit vom eigentlich Fach-Code sein kann. Das kann sinnbildlich am anderen Ende der Anwendung sein, dass da Threads gestartet werden oder es sind einfach Threads vom Framework. Sprich, man hat in der Regel am Code nie einen Marker "Wird Multi-Threaded ausgeführt".

Analyse verändert das Verhalten
Multi-Threading Bugs sind Timing Probleme. Wie man hier im Thread schön sieht, treten die nicht immer auf. Wenn ich mit Debugger rangehe verändere ich das Timing der Anwendung extrem. Das heißt, viele Threading Probleme tauchen nie auf, wenn ich mit dem Debugger mir das Verhalten ansehe. Mit etwas Glück kann man mit Logging / System.out.println was erreichen - aber selbst das kann das Timing Verhalten verändern.
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
Leyla Thread isInterrupt Java Basics - Anfänger-Themen 18
P Meldung aus Java-Klasse in Thread an aufrufende Klasse Java Basics - Anfänger-Themen 1
A Thread XML-Dateien zusammenfügen Java Basics - Anfänger-Themen 11
F influxdb Upload in eigenem Thread Java Basics - Anfänger-Themen 2
frager2345 Thread - Methoden synchronized deklarieren Java Basics - Anfänger-Themen 10
berserkerdq2 Größter unterschied von extends thread und implements runnable? Java Basics - Anfänger-Themen 2
T Thread beenden aus zweiter Klasse Java Basics - Anfänger-Themen 4
A Thread - Synchronized Java Basics - Anfänger-Themen 10
A Thread Producer - Consumer Java Basics - Anfänger-Themen 1
A Thread-Semhapore Java Basics - Anfänger-Themen 0
A Thread Exchanger Java Basics - Anfänger-Themen 22
A Thread-Cyclicbarrier Java Basics - Anfänger-Themen 4
B In einem Thread Endlosschleife beenden Java Basics - Anfänger-Themen 19
A Thread-Verklemmung Java Basics - Anfänger-Themen 10
A Thread find number Java Basics - Anfänger-Themen 8
F Thread.sleep() Java Basics - Anfänger-Themen 5
F Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 11 at main.main(main.java:11) Java Basics - Anfänger-Themen 2
A Thread Java Basics - Anfänger-Themen 3
M Exception in thread "main" java.util.NoSuchElementException Java Basics - Anfänger-Themen 2
A Thread Java Basics - Anfänger-Themen 8
B Compiler-Fehler Fehlermeldung Exception in thread, falsche Eingabewert Java Basics - Anfänger-Themen 2
M Thread-Zustände Java Basics - Anfänger-Themen 6
CptK For-Schleife in Thread nach jedem Durchlauf pausieren Java Basics - Anfänger-Themen 35
S Kriege Fehler "Exception in thread" beim Benutzen von SubStrings. Java Basics - Anfänger-Themen 2
B Endlosschleife Thread sauber beenden Java Basics - Anfänger-Themen 19
D Java Thread wartet nur ein mal Java Basics - Anfänger-Themen 1
D Java Thread wartet nur ein mal Java Basics - Anfänger-Themen 0
O Exception in thread "main" java.lang.ArithmeticException: / by zero Java Basics - Anfänger-Themen 4
C Thread und TimerTask, Verstädnisproblem Java Basics - Anfänger-Themen 10
amgadalghabra Sorting Thread Launcher Java Basics - Anfänger-Themen 3
B Exception in thread "AWT-EventQueue-0" java.util.ConcurrentModificationException Java Basics - Anfänger-Themen 8
A Thread Java Basics - Anfänger-Themen 4
A Thread Java Basics - Anfänger-Themen 1
A Thread Java Basics - Anfänger-Themen 0
R Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException Java Basics - Anfänger-Themen 5
S Compiler-Fehler Exception in thread "main" java.lang.Error: Unresolved compilation problem: Java Basics - Anfänger-Themen 6
L Liste in anderem Thread laden Java Basics - Anfänger-Themen 1
B Thread / Prozess stoppen? Java Basics - Anfänger-Themen 22
I Compiler-Fehler Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 5 Java Basics - Anfänger-Themen 3
B Threads Thread sleep() Method einfache Frage Java Basics - Anfänger-Themen 8
W Thread Aufgabe - Vorgehensweise Java Basics - Anfänger-Themen 8
L Liste in anderem Thread laden Java Basics - Anfänger-Themen 0
J Threads PrograssBar update während thread Java Basics - Anfänger-Themen 13
D Compiler-Fehler Wert auf Datenbank übertragen und Sleep Thread Java Basics - Anfänger-Themen 3
Spencer Reid JavaFX Memory Thread.sleep Java Basics - Anfänger-Themen 1
S Thread.sleep mit JProgressBar Java Basics - Anfänger-Themen 1
ralfb1105 Frage zu Thread Synchronisation mit wait() und notify() Java Basics - Anfänger-Themen 3
R Exception in thread "main" java.lang.NullPointerException Java Basics - Anfänger-Themen 10
J JavaFX -> SocketIO -> Thread -> Update Label Java Basics - Anfänger-Themen 13
J Thread Handling Java Basics - Anfänger-Themen 9
A Problem mit Thread.sleep Java Basics - Anfänger-Themen 4
C Thread in Methode + raus aus der Schleife Java Basics - Anfänger-Themen 10
E Threads Thread in While-Schleife nur einmal starten Java Basics - Anfänger-Themen 2
F Daten von Thread an den aufrufenden zurückgeben Java Basics - Anfänger-Themen 22
C Compiler-Fehler Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 2 Java Basics - Anfänger-Themen 3
B Thread Problem Java Basics - Anfänger-Themen 7
N KeyListener in Thread Java Basics - Anfänger-Themen 0
M Thread.sleep() Funktion Java Basics - Anfänger-Themen 1
W JLabel in Main aus Thread verändern. Java Basics - Anfänger-Themen 4
D Ausgeben welcher Thread gerade Arbeitet Java Basics - Anfänger-Themen 8
N Threads Thread-Fehler Java Basics - Anfänger-Themen 2
F Thread um Uhrzeit ausführen Java Basics - Anfänger-Themen 5
F Get/Post als eigener Thread mit Rückgabe Java Basics - Anfänger-Themen 5
J Exception in thread "main" Java Basics - Anfänger-Themen 1
F Thread der auf eine Queue wartet, sicher beenden Java Basics - Anfänger-Themen 4
B Animation mit Thread(s) Java Basics - Anfänger-Themen 23
I Thread.sleep (1000); Java Basics - Anfänger-Themen 1
M Threads Jede Klasse einem Thread zuweisen Java Basics - Anfänger-Themen 7
J Java Thread cancel() und wiederbeleben Java Basics - Anfänger-Themen 4
J BouncingBalls 1 Thread Java Basics - Anfänger-Themen 3
L Fehler: Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException Java Basics - Anfänger-Themen 4
J Timer oder Thread programmieren ? Java Basics - Anfänger-Themen 10
fLooojava Laufender Thread | Boolean ändern Java Basics - Anfänger-Themen 9
T Thread Pool mit Work Stealing Java Basics - Anfänger-Themen 1
R Java Thread Java Basics - Anfänger-Themen 10
J Welche Methoden laufen im neuen thread ?? Java Basics - Anfänger-Themen 9
S Java memory fehler: Exception in thread "AWT-EventQueue-0" java.lang.OutOfMemoryError: Java heap spa Java Basics - Anfänger-Themen 5
K Thread - Methoden in die run Methode Schreiben Java Basics - Anfänger-Themen 5
N Threads Exception in thread "main"... Feher bei dem Versuch ein Radius zu berechnen Java Basics - Anfänger-Themen 4
A Code läuft nicht, Fehlermeldung Exception in thread "main" java.lang.Error: Unresolved compilation " Java Basics - Anfänger-Themen 11
V Threads Exception in Thread behandeln Java Basics - Anfänger-Themen 3
S Methoden Multi-Thread und Methoden Objects. Java Basics - Anfänger-Themen 1
J Thread erstellen (BlueJ Projekt) Java Basics - Anfänger-Themen 3
P Exception in thread "main" java.lang.NoClassDefFoundError: Java Basics - Anfänger-Themen 1
F Threads Variable aus einem Thread in main Methode? Java Basics - Anfänger-Themen 9
K Exception in thread "main" Java Basics - Anfänger-Themen 7
L Thread-Frage Java Basics - Anfänger-Themen 2
E Was ist ein idle-thread? Java Basics - Anfänger-Themen 1
D Exception in thread "AWT-EventQueue-0" Java Basics - Anfänger-Themen 8
J Threads Prozess in Thread auslagern Java Basics - Anfänger-Themen 2
G Thread mehrmals starten und schliessen Java Basics - Anfänger-Themen 6
F Thread Koordination (Vorteile/Nachteile) Java Basics - Anfänger-Themen 0
O Thread aus dem Thread stoppen Java Basics - Anfänger-Themen 6
O Swingworker/Thread Java Basics - Anfänger-Themen 3
R Focus auf JPanel im Thread Java Basics - Anfänger-Themen 9
S musik in eigenem thread Java Basics - Anfänger-Themen 2
A Klasse,Vererbung,Interface,Singleton,Thread Java Basics - Anfänger-Themen 5
IngoF GUI mit Thread Daten austauschen. Java Basics - Anfänger-Themen 6
L Compiler-Fehler Exception in thread "main" java.lang.NullPointerException Java Basics - Anfänger-Themen 2
F Exception in thread main java.lang.StackOverflowError Java Basics - Anfänger-Themen 3

Ähnliche Java Themen

Neue Themen


Oben