During our lessons in the university, we learned about Threads
and used the "Busy Waiting" method for an example of a Car
waiting at a TrafficLight
. For this task we build three classes:
TrafficLight (implements Runnable)
Car (implements Runnable)
Main
In our Main
class we start two Thread
s, one of Car
, and one of TrafficLight
. The Car
has the boolean attribute hasToWait
. The run()
method in this class works the way, that it works through a while
loop as long as hasToWait == true
. To change this, we have the notifyCar()
method in the Car
class, which is used by the TrafficLight
. The run()
method in TrafficLight
runs through a Thread.sleep()
to simulate a certain time of waiting.
Everything works fine at my Prof's but eventually I have serious problems with it. As long as the while
loop in the Car
class is empty. When I put in a System.out.println()
- which is not empty, it works. But if the Syso is empty, the result is no displaying of the Text of the Run
method.
Also it's working when the Thread.sleep()
in TrafficLight
is 0
. Than it works with an empty while
loop.
Here is my code:
package trafficlight;
public class Car implements Runnable {
private boolean hasToWait = true;
public void run() {
this.crossTrafficLight();
}
public void crossTrafficLight() {
while(hasToWait){ for(int i = 0; i<20; i++){System.out.println("123");}} // Busy waiting
System.out.println("Auto fährt über Ampel");
}
public void notifyCar() {
this.hasToWait = false;
System.out.println("Test");
}
}
package trafficlight;
public class TrafficLight implements Runnable {
private Car car;
public TrafficLight(Car car) {
this.car = car;
}
@Override
public void run() {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
this.car.notifyCar();
}
}
package trafficlight;
public class Main {
public static void main(String[] args){
Car car = new Car();
TrafficLight tl = new TrafficLight(car);
new Thread(car).start();
new Thread(tl).start();
}
}
Where is the problem? Why does it work at my profs but not at my computer? I got the code 1:1 in my Eclipse Juno, using JRE 1.7
In addition to everything said in this other answer (just substitute your hasToWait
for finished
in that answer), the reason why the code starts working when you add a println
is as follows:
println
is a synchronized method;You could say that it starts working mostly by accident: you are piggybacking on the synchronization going on in println
.
The real problem with your code is the instance field hasToWait
. This field is being used by two threads. The car thread reads the value, and the traffic light thread updates the value after some time.
The access to this field must be synchronized in some way.
There are two ways to do this:
Use the synchronized
keyword. Either by using a synchronized block at all places, where it is read or written, or - better - write a synchronized getter and a synchronized setter, then use the getter and the setter inside the Car class.
Use the volatile
keyword. Just declare your field as volatile. This keyword exists for exactly that case. More information on volatile can be found in Oracle's Java Tutorials.
After reading the article about atomic access (see link above), it should be clear that option 2 (declaring volatile) is the far better option - for this use case.
Now to the difference you see between your computer and your professor's computer: As long as you are using a single-core-processor, you will see updates on an instance field in other threads as though they were synchronized, because the CPU does not have to synchronize these values in the other cores' cache areas. If you use a multi-core-processor, then the JVM is able to run threads on several cores. That means, that these cores have to synchronize values, and the volatile mechanism is exactly designed for that.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With