Frage bzgl. Multithreading

Klösp

Aktives Mitglied
Hallo,

wir hatte hier die Tage mal ein Thema wo es um synchronized-Blöcke ging.
Hatte mich im nachher noch etwas damit beschäftigt und mir ist eine Frage dazu gekommen.

Hab das Beispiel nochmal sinngemäß nachgebaut:

Java:
public class Ablage {
	
	int zettelAnzahl;

	public Ablage(int zettelAnzahl) {
		super();
		this.zettelAnzahl = zettelAnzahl;
	}

	public int getZettelAnzahl() {
		return zettelAnzahl;
	}

	public void setZettelAnzahl(int zettelAnzahl) {
		this.zettelAnzahl = zettelAnzahl;
	}

}


Java:
public class Arbeiter extends Thread {

	Betrieb meinBetrieb;

	public Arbeiter(Betrieb meinBetrieb) {
		super();
		this.meinBetrieb = meinBetrieb;
	}

	public Betrieb getMeinBetrieb() {
		return meinBetrieb;
	}

	public void setMeinBetrieb(Betrieb meinBetrieb) {
		this.meinBetrieb = meinBetrieb;
	}

	@Override
	public void run() {
        while( !isInterrupted() )
        {
            meinBetrieb.nehmeZettel(10);
            meinBetrieb.legeZettelab(10);
            try
            {
                Thread.sleep(100);
            }
            catch(InterruptedException e)
            {
                interrupt();
            }
        }
	}

}

Java:
package syncBeispiel;

public class Betrieb {
	
	Ablage[] ablagen;
	int anzahlAblagen;

	public Betrieb(int anzahlAblagen, int zettelpAblage) {
		super();
		this.anzahlAblagen=anzahlAblagen;
		this.ablagen = new Ablage[anzahlAblagen];
		for(int i=0; i<ablagen.length;i++){
			ablagen[i]=new Ablage(zettelpAblage);
		}	
	}
	
	public void nehmeZettel(int anzahl){
		Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
		p.setZettelAnzahl(p.getZettelAnzahl()-anzahl);
	}
	
	
	public void legeZettelab(int anzahl){
		Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
		p.setZettelAnzahl(p.getZettelAnzahl()+anzahl);
	}
	
	public void pruefen(){
		int sum=0;
		for(Ablage p:ablagen){
			sum+=p.getZettelAnzahl();
		}
		
		System.out.println(sum);	
	}
	
}

Java:
public class SyncTest {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		Betrieb betrieb = new Betrieb(5, 45);
		Arbeiter[] arbeiter = new Arbeiter[50];

		for (int i = 0; i < arbeiter.length; i++) {
			arbeiter[i] = new Arbeiter(betrieb);
			arbeiter[i].start();
		}

		try {
			Thread.sleep(60000);
		} catch (InterruptedException e) {
		}
		
		 for(int i = 0; i < 50; i++)
	        {
	            arbeiter[i].interrupt();
	            try {
					arbeiter[i].join();
				} catch (InterruptedException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
	        }
		 
		 betrieb.pruefen();

	}

}

Nun soll das Beispiel also einen typischen Büro-Alltag simulieren. Es werden immer ein paar Zettel (10) von einer beliebigen Ablage genommen und dann auf irgendeine andere Ablage verschoben.
Es sollten natürlich keine Zettel abhanden kommen (vieleicht doch nicht so realistisch^^).

Möglichkeit1:
Die Betriebsmethoden werden komplett synchronisiert.
Java:
	public synchronized void  nehmeZettel(int anzahl){
		Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
		p.setZettelAnzahl(p.getZettelAnzahl()-anzahl);
	}
	
	
	public synchronized  void legeZettelab(int anzahl){
		Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
		p.setZettelAnzahl(p.getZettelAnzahl()+anzahl);
	}
Funktioniert, ist aber nicht gut, weil wenn ein Arbeiter eine Aktion durchführt ist immer der ganze Betrieb lahmgelegt.


Möglichkeit 2:
Java:
	public void  nehmeZettel(int anzahl){
		Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
		synchronized(p){
		p.setZettelAnzahl(p.getZettelAnzahl()-anzahl);
		}
	}
	
	
	public void legeZettelab(int anzahl){
		Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
		synchronized(p){
		p.setZettelAnzahl(p.getZettelAnzahl()+anzahl);
		}
	}
Sollte besser sein und funktioniert auch soweit.
Allerdings hatte ich zuerst die getter/setter Methoden von Ablage synchronisiert.

Java:
	public synchonized  int getZettelAnzahl() {
		return zettelAnzahl;
	}

	public synchronized void setZettelAnzahl(int zettelAnzahl) {
		this.zettelAnzahl = zettelAnzahl;
	}
Das hat dann allerdings nicht funktioniert.
Woran liegt das?

Meine Vermutung:
Würde man den Aufruf der getter/setter so schreiben ist das Problem offensichtlich.
Java:
	public void nehmeZettel(int anzahl) {
		Ablage p = ablagen[(int) (Math.random() * this.anzahlAblagen)];
		int number = p.getZettelAnzahl();
		p.setZettelAnzahl(number - anzahl);

	}

	public void legeZettelab(int anzahl) {
		Ablage p = ablagen[(int) (Math.random() * this.anzahlAblagen)];
			int number=p.getZettelAnzahl();
			p.setZettelAnzahl(number + anzahl);
	
	}

Ist also dieser Aufruf das gleiche wie oben?
Java:
p.setZettelAnzahl(p.getZettelAnzahl() + anzahl);
Was macht der Compiler daraus, bzw wie wird das zur Laufzeit ausgeführt?
Wann hat der aufrufende Thread den Monitor auf die Ablage und wann wird er freigegeben?

Ist es der Grund für das Problem,dass der Monitor trotzdem zwischen get und set nochmal freigegeben werden kann, oder übersehe ich noch etwas anderes?

Ich hoffe es ist einigermaßen klar geworden was ich wissen möchte. Ist etwas lang geworden.

Vielen Dank im Vorraus
 

Klösp

Aktives Mitglied
Das wäre auch eine Möglichkeit.
Meine Frage war aber mehr wieso es so funktioniert,

Java:
    public void  nehmeZettel(int anzahl){
            Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
            synchronized(p){
            p.setZettelAnzahl(p.getZettelAnzahl()-anzahl);
            }
        }
       
       
        public void legeZettelab(int anzahl){
            Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
            synchronized(p){
            p.setZettelAnzahl(p.getZettelAnzahl()+anzahl);
            }
        }

mit:
Java:
    public int getZettelAnzahl() {
            return zettelAnzahl;
        }
     
        public void setZettelAnzahl(int zettelAnzahl) {
            this.zettelAnzahl = zettelAnzahl;
        }

und so nicht.


Java:
    public void  nehmeZettel(int anzahl){
            Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
            p.setZettelAnzahl(p.getZettelAnzahl()-anzahl);
        }
       
       
        public void legeZettelab(int anzahl){
            Ablage p=ablagen[(int) (Math.random()*this.anzahlAblagen)];
            p.setZettelAnzahl(p.getZettelAnzahl()+anzahl);
        }


mit:
Java:
    public synchonized  int getZettelAnzahl() {
            return zettelAnzahl;
        }
     
        public synchronized void setZettelAnzahl(int zettelAnzahl) {
            this.zettelAnzahl = zettelAnzahl;
        }
 
Zuletzt bearbeitet:

wef34fewrg

Aktives Mitglied
hey ho,

das sieht ja aus wie das Bankenbeispiel von ein paar Tagen, nicht wahr? :D Wieso hast du denn nicht gleich da die Frage reingeschrieben?

Erst mal. Falls du keinen bock hast zu warten. Wenn du mal die sleep Methode von 60000 auf 6000 reduzierst, wirst du die Ergebnisse schon sehen. ;)


Was deine Annahme angeht hast du recht.
Die Sperre wird beim Aufruf bzw. beim Verlassen der Methoden gesetzt bzw. aufgehoben. Rufst du also p.setZettelAnzahl(p.getZettelAnzahl()-anzahl); auf, wird die Sperre beim get Aufruf gesetzt und beim Verlassen wieder gelöst. Nachdem die Addition vollzogen wurde, wird die Sperre erst beim set Aufruf wieder gesetzt. In der Zeit kann sich der Kontostand natürlich verändern.



Java:
public void nehmeZettel(int anzahl) {
        Ablage p = ablagen[(int) (Math.random() * this.anzahlAblagen)];
        int number = p.getZettelAnzahl();
        p.setZettelAnzahl(number - anzahl);
 
    }

In der Methode kannst du das wunderbar nachvollziehen. Wenn zwei Threads nacheinander die Zeile int number = p.getZettelAnzahl(); aufrufen (z.B. mit dem Ergebnis 20), dann eine Pause einlegen und weiter machen, führt die Subtraktion in beiden Fällen zum Wert 10.Das heisst, 10 Zettel gehen dabei verloren.

Übrigens müsste das Vertauschen der Zeilen
meinBetrieb.nehmeZettel(10);
meinBetrieb.legeZettelab(10);
zu anderen Ergebnissen führen, als die, die du jetzt so bekommst. Wenn du Spaß am Knobeln hast, dann frag dich mal warum das so ist.

P.S. ich hoffe da kommen auch wirklich die Ergebnisse raus, an die ich denke (+/-) :D



Gruß
 

Klösp

Aktives Mitglied
Hallo,

vielen Dank für die Hilfe.

Was genau meinst du, was sich verändern soll.
Die Summe bei der unsynchronisierten Version, oder in der synchronsierten Fassung die Anzahl der Zettel auf einer Ablage?
und gib mal nen Tipp...:)

mfg.
 

Klösp

Aktives Mitglied
Nochmal eine kleine Ergänzung.

Hab noch ein wenig dran rumgespielt, weil ich das ganze mal mit Locks probieren wollte.
Dazu hab ich das ganze etwas umgebaut.
Das Zettel verschieben macht jetzt der Arbeiter.
Java:
import java.util.Random;

public class Arbeiter extends Thread {

	int durchläufe = 0;
	static Random r = new Random();
	Betrieb meinBetrieb;

	public Arbeiter(Betrieb meinBetrieb) {
		super();
		this.meinBetrieb = meinBetrieb;
	}

	public Betrieb getMeinBetrieb() {
		return meinBetrieb;
	}

	public void setMeinBetrieb(Betrieb meinBetrieb) {
		this.meinBetrieb = meinBetrieb;
	}

	public void verschieben(int anzahlZettel) {
		int anzahlAblagen = meinBetrieb.ablagen.length;
		Ablage von = meinBetrieb.ablagen[r.nextInt(anzahlAblagen)];
		Ablage nach = meinBetrieb.ablagen[r.nextInt(anzahlAblagen)];
		boolean b = von.neuberechnen(-anzahlZettel);
		if (b) {
			nach.neuberechnen(anzahlZettel);
		}
	}

	@Override
	public void run() {
		while (!isInterrupted()) {
			// meinBetrieb.nehmeZettel(10);
			// meinBetrieb.legeZettelab(10);
			this.verschieben(10);
			durchläufe++;

			try {
				Thread.sleep(100);
			} catch (InterruptedException e) {
				interrupt();
			}
		}
	}
}

Damit ich das mit Locks machen kann muss ich die eigentliche Berechnung in der Ablage machen, weil ich sonst wieder das alte Problem habe.
Die Möglichkeit ein Lock auf ein bestimmtes anderes Objekt zubekommen wie bei synchronized(object) wird es ja vermutlicht geben, oder?
Java:
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock;

public class Ablage {

	private int zettelAnzahl;
	private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
	private final Lock readLock = lock.readLock(),
			writeLock = lock.writeLock();

	public Ablage(int zettelAnzahl) {
		super();
		this.zettelAnzahl = zettelAnzahl;
	}

	public int getZettelAnzahl() {
		try {
			readLock.lock();
			return zettelAnzahl;
		} finally {
			readLock.unlock();
		}
	}

	public void setZettelAnzahl(int zettelAnzahl) {
		try {
			writeLock.lock();
			this.zettelAnzahl = zettelAnzahl;
		} finally {
			writeLock.unlock();
		}
	}

	public boolean neuberechnen(int aenderung) {
		try {
			writeLock.lock();
			int check;
			if ((check = zettelAnzahl + aenderung) >= 0) {
				zettelAnzahl = check;
				return true;
			}
			return false;
		} finally {
			writeLock.unlock();
		}
	}
}
Zumindestens im setter muss ich auch noch locken um sicher zu gehen, dass nicht von irgendwo außerhalb dieses Beispiel einer "dazwischen funken" kann.
Dabei kam bei mir eine Frage auf.
Ist Monitor gleich Monitor? Würde es funktionieren wenn ich in der neuberechnen-Methode weiter den Lock nutze und den setter synchronized deklariere?

mfg
 
Zuletzt bearbeitet:
Ähnliche Java Themen
  Titel Forum Antworten Datum
D Frage bzgl. Enum-Handhabung Java Basics - Anfänger-Themen 16
D Frage bzgl. Interfaces Java Basics - Anfänger-Themen 10
M Frage bzgl. Testbarkeit von Methoden in Relation zu Ihren Abhängigkeiten Java Basics - Anfänger-Themen 11
M Frage bzgl. Oracle Java Zertifizierung Java Basics - Anfänger-Themen 4
1 Frage bzgl. Polymorphismus Java Basics - Anfänger-Themen 6
X Frage zur Vererbung (bzgl Supercontruktor) Java Basics - Anfänger-Themen 7
L Frage bzgl. Threads beim Bilder laden Java Basics - Anfänger-Themen 3
P Frage bzgl. Class Sharing Java Basics - Anfänger-Themen 2
G Dringende Frage bzgl. meines Codes Java Basics - Anfänger-Themen 30
C Frage bzgl. main Java Basics - Anfänger-Themen 2
C Frage bzgl. action von Button Java Basics - Anfänger-Themen 6
Zrebna Frage zu Test-Driven Development (TDD) Java Basics - Anfänger-Themen 3
I Frage Thymeleaf -> Fehler ignorieren und mit "" ersetzen? Java Basics - Anfänger-Themen 15
I Frage Thymeleaf -> Prefix / Suffix ändern? Java Basics - Anfänger-Themen 11
D Rekursions Probleme / frage Java Basics - Anfänger-Themen 4
T Frage zu Parse Java Basics - Anfänger-Themen 2
H Frage an die Profis Java Basics - Anfänger-Themen 4
J Eine konzeptionelle Frage zu OOP Java Basics - Anfänger-Themen 3
P Frage zu Rekursion und Backtracking Java Basics - Anfänger-Themen 2
H Frage zur Ausgabe Java Basics - Anfänger-Themen 4
H Frage zu arithmetischen Operationen Java Basics - Anfänger-Themen 20
F Kurze Frage zu replace() Java Basics - Anfänger-Themen 19
JavaSchmecktLecker Polymorphie Frage zur Methodenüberschreibung Java Basics - Anfänger-Themen 21
J Frage zu einem "Taschenrechner" code Java Basics - Anfänger-Themen 9
B Erste Schritte Frage zu Instanzierung und Referenzen Java Basics - Anfänger-Themen 8
DoubleM Runtime.getRuntime().exec Frage Java Basics - Anfänger-Themen 2
J Eine theoretische Frage zur Praxis - JPanel oder Canvas Java Basics - Anfänger-Themen 5
O Frage: Formaler Typbezeichner? Java Basics - Anfänger-Themen 3
I BlueJ Queue Frage für Klausur Java Basics - Anfänger-Themen 2
N Verständnis Frage zu Variablen Java Basics - Anfänger-Themen 3
N Spezielle frage zum Comparator Java Basics - Anfänger-Themen 6
L Frage zum Array Java Basics - Anfänger-Themen 1
A Frage zum UML Design Java Basics - Anfänger-Themen 1
I Hilfe bei Klausur Frage Java Basics - Anfänger-Themen 8
izoards Drucken Frage zu FAQ Beitrag Java Basics - Anfänger-Themen 2
J Frage zu meinem Code (OOP) Java Basics - Anfänger-Themen 4
sserio Split() -> Regex Frage. Java Basics - Anfänger-Themen 7
A OCA Study Guide: 2. Frage aus Kapitel 3 Java Basics - Anfänger-Themen 9
sserio Date Library Frage Java Basics - Anfänger-Themen 9
Max246Sch Frage zu Währungsrechner Code Java Basics - Anfänger-Themen 2
sserio Frage zu HashMaps Java Basics - Anfänger-Themen 20
sserio Frage zu Threading - Multithreading Java Basics - Anfänger-Themen 2
sserio Frage zu Lambda Ausdrücken Java Basics - Anfänger-Themen 7
sserio Frage zu BigInteger Java Basics - Anfänger-Themen 1
xxx12 Frage Java Basics - Anfänger-Themen 2
I Generelle Frage zu Mikroservices (Spring Boot?), Docker... Java Basics - Anfänger-Themen 7
R Frage zu Methoden (Rückgabewert u. ohne.) Java Basics - Anfänger-Themen 2
A Frage zur programmierung Java Basics - Anfänger-Themen 12
M Frage zur Methode split der Klasse String Java Basics - Anfänger-Themen 32
R Input/Output Frage zu Java IO Java Basics - Anfänger-Themen 6
M Frage zu printWriter Java Basics - Anfänger-Themen 5
C Frage zu OLSMultipleLinearRegression Java Basics - Anfänger-Themen 31
KogoroMori21 Frage zum Euklidischen Algorithmus Java Basics - Anfänger-Themen 11
S Verständnis-Frage zu einer HÜ? Java Basics - Anfänger-Themen 1
F Frage betreff Programm mit dem man C++-Code in JAVA-Code übersetzen lassen kann Java Basics - Anfänger-Themen 2
L Frage zur Ticket Maschine Java Basics - Anfänger-Themen 1
J Frage zu OOP-Klassendiagramm Java Basics - Anfänger-Themen 8
OSchriever Frage zu Compiler Java Basics - Anfänger-Themen 8
H Frage zu Throw Exception Java Basics - Anfänger-Themen 2
TimoN11 Frage zu Java-Vererbung (Cast) Java Basics - Anfänger-Themen 5
Bademeister007 Hallo Leute ich hab eine Frage zur ArrayList Java Basics - Anfänger-Themen 8
F Frage betreff Programmierbücher zu Lagerverwaltung als Konsolenprogramm Java Basics - Anfänger-Themen 3
dieter000 Kurze Frage kann mir ejmand kurz diesen Code erklären, bzw wie man die zeilen erklärt und so Java Basics - Anfänger-Themen 1
I String.split regex Frage Java Basics - Anfänger-Themen 2
N Best Practice Frage zum MVC-Pattern Java Basics - Anfänger-Themen 2
dieter000 Frage zu einem Beispiel... Java Basics - Anfänger-Themen 5
J Frage zum Loggen Java Basics - Anfänger-Themen 18
J Methoden Frage: Array-Werte in anderer Methode ändern Java Basics - Anfänger-Themen 4
Zrebna Frage zum "Referenzen-konzept" in Java Java Basics - Anfänger-Themen 8
JD_1998 Array-Position aus einer Methode in einer anderen ausgeben (Kurze Frage) Java Basics - Anfänger-Themen 2
marcooooo Frage zu bestimmten Beispiel Java Basics - Anfänger-Themen 31
NeoLexx equals()-Methode Verständnis Frage anhand Code Beispiel Java Basics - Anfänger-Themen 22
N Input/Output Eine Frage über system.out.println. Java Basics - Anfänger-Themen 10
B Erste Schritte Learning Coding (!) Frage an erfahrene Programmierer. Java Basics - Anfänger-Themen 23
M konzeptuelle Frage: In welcher Klasse definiert man am Besten Methoden, die die Kommunikation mit dem User regeln? Java Basics - Anfänger-Themen 8
B Frage zum Code verständnis im Resultat Java Basics - Anfänger-Themen 10
C Exception-Frage Java Basics - Anfänger-Themen 3
J Eine Frage zur Schreibweise == ? : Java Basics - Anfänger-Themen 3
S Frage des Designs Java Basics - Anfänger-Themen 1
JavaTalksToMe Extends/Implements Frage Java Basics - Anfänger-Themen 3
pkm Frage zu Servletfunktion Java Basics - Anfänger-Themen 0
B Frage zur Währungsumrechnung Java Basics - Anfänger-Themen 3
S Allgemeine Frage über Generics und Vererbungen Java Basics - Anfänger-Themen 5
Kirby.exe Frage zur Verwendung von Interfaces Java Basics - Anfänger-Themen 6
D Frage zu Strings einer Exception Java Basics - Anfänger-Themen 4
L Wie frage ich ab, ob in einem Array, Werte doppelt vorkommen? Java Basics - Anfänger-Themen 4
D Frage zur IDE IntelliJ IDEA Java Basics - Anfänger-Themen 6
H Frage zum 2d Array Java Basics - Anfänger-Themen 1
N Frage zum Newton-Fraktal Java Basics - Anfänger-Themen 1
H Frage zu interfaces Java Basics - Anfänger-Themen 1
J Frage dazu Variablen klassenübergreifend zu verändern Java Basics - Anfänger-Themen 22
I Frage zu SkipList Java Basics - Anfänger-Themen 4
G Frage zu JScrollPane Java Basics - Anfänger-Themen 12
Kirby.exe Allgemeine Frage Java Basics - Anfänger-Themen 3
W Frage zu anonymen Klassen Java Basics - Anfänger-Themen 4
J Kleine Frage zu OOP Java Basics - Anfänger-Themen 371
S Frage Klasse und Objekte Java Basics - Anfänger-Themen 2
F Frage zu Iteratoren Java Basics - Anfänger-Themen 2
C Erste Schritte Frage zur ArrayList Java Basics - Anfänger-Themen 15
J Frage zur Vererbung Java Basics - Anfänger-Themen 1

Ähnliche Java Themen

Neue Themen


Oben