|
|
Doing Two or More Tasks At Once: Threads |
The code segments within a program that access the same object from separate, concurrent threads are called critical sections. In the Java language, a critical section can be a block or a method and are identified with thesynchronizedkeyword. The Java platform then associates a lock with every object that has synchronized code.In the producer/consumer example, the
putandgetmethods of theCubbyHole(page 645) are the critical sections. TheConsumershould not access theCubbyHolewhen theProduceris changing it, and theProducershould not modify it when theConsumeris getting the value. Soputandgetin theCubbyHoleclass should be marked with thesynchronizedkeyword.Here's a code skeleton for the
CubbyHoleclass:Note that the method declarations for bothpublic class CubbyHole { private int contents; private boolean available = false; public synchronized int get() { ... } public synchronized void put(int value) { ... } }putandgetcontain thesynchronizedkeyword. Hence, the system associates a unique lock with every instance ofCubbyHole(including the one shared by theProducerand theConsumer). Whenever control enters a synchronized method, the thread that called the method locks the object whose method has been called. Other threads cannot call a synchronized method on the same object until the object is unlocked.So, when the
ProducercallsCubbyHole'sputmethod, it locks theCubbyHole, thereby preventing theConsumerfrom calling theCubbyHole'sgetmethod:When thepublic synchronized void put(int value) { // CubbyHole locked by the Producer .. // CubbyHole unlocked by the Producer }putmethod returns, theProducerunlocks theCubbyHole.Similarly, when the
ConsumercallsCubbyHole'sgetmethod, it locks theCubbyHole, thereby preventing theProducerfrom callingput:The acquisition and release of a lock is done automatically and atomically by the Java runtime system. This ensures that race conditions cannot occur in the underlying implementation of the threads, thus ensuring data integrity. Synchronization isn't the whole story. The two threads must also be able to notify one another when they've done their job. Learn more about that after a brief foray into reentrant locks.public synchronized int get() { // CubbyHole locked by the Consumer ... // CubbyHole unlocked by the Consumer }
|
|
Doing Two or More Tasks At Once: Threads |