How to Prevent Race Conditions in Java 8

See Java: Tips and Tricks for similar articles.
Looking to improve your Java skills? Check out Webucator’s Java classes.

A race condition occurs in programming when two or more execution threads modify a shared, or critical, resource. Race conditions can result in run time errors that are difficult to isolate and to repair. The term "race" is used because the threads can be regarded as racing each other to complete operations on a variable or other shared resource.

In Java 8, race conditions can be prevented by enforcing single threading through methods that modify shared resources. To learn how to prevent race conditions in Java 8, follow these eight steps.

  1. Open your text editor and create the Java program that will demonstrate the race condition. Type in the following Java statements:
  2. public class PreventRaceConditions implements Runnable {
    	private int inventory=1;
    	long twoSeconds=2000;
    	public static void main (String[] args) {
    		PreventRaceConditions prc=new PreventRaceConditions();
    		Thread firstThread=new Thread(prc, "First thread");
    		Thread secondThread=new Thread(prc, "Second thread");
    		firstThread.start();
    		secondThread.start();
    	}
    	public void run() {
    		if (inventory > 0) {
    			try {
    				System.out.println("Thread " + Thread.currentThread().getName() + " starting with inventory: " + inventory);
    				Thread.sleep(twoSeconds);
    			}
    			catch (InterruptedException e) {
    					// no op
    			}	
    			inventory--;
    			System.out.println("Thread " + Thread.currentThread().getName() + " leaving with inventory: " + inventory);
    		}
    		else {
    			System.out.println("Need to backorder! Thread " + Thread.currentThread().getName() + " found inventory with: " + inventory);
    		}
    	}	
    }

    The program defines an inventory count for an item in our inventory. The variable will serve as our shared resource. The variable is initialized to a value of 1. The program will process orders for this item given that at least one item is present in our inventory. To create a race condition for this critical resource, two threads will be started. Each thread will check the inventory count to ensure that an order can be filled. If the order can be filled then the inventory count will be decremented. If not, then the program indicates that item should be back-ordered. Each thread will be paused 2 seconds so that the other thread can execute. By doing so we are simulating a real world multi-threaded environment.

    The problem occurs when a thread checks the inventory count and decides the order can be filled but is then swapped out, allowing another thread to execute the run method. The second thread will also decide that the order can be filled when in fact the first thread will soon decrement the inventory count, putting this item out of stock.

    You will compile and run the program so that you witness the race condition. Then you will edit the program to fix the race condition.

  3. Save your file as PreventRaceConditions.java.
  4. Open a command prompt and navigate to the directory containing your new Java program. Then type in the command to compile the source and hit Enter.Compile Program with Race Condition
  5. You are ready to test your Java program. Type in the command to run the Java runtime launcher and hit Enter.Run Program with Race ConditionNotice that each thread determined that the inventory count was greater than 0 and therefore the order could be filled. But upon leaving the method, each thread displays an inventory count of -1. Obviously, an order cannot be shipped with 0 items and the operation should not be attempted in the first place. The problem is caused by a race condition. In the next step, you will modify the program to eliminate the race condition.
  6. Return to your edit session. You will add two statements to the program to solve the race condition. I have highlighted the statements you need to add to the program in the image below:Java Source for Race ConditionThe synchronized keyword declares that only one thread can execute the code contained within the block that is delimited by the curly braces. The this keyword indicates that a lock will be placed on the object instance while a thread controls execution of the statements in the block.
  7. Save your changes.
  8. In the command prompt, type in the command to compile the source and hit Enter.Compile Program with Race Condition
  9. You are ready to test your changes. Type in the command to run the Java runtime launcher and hit Enter.Run Program with Race ConditionThe first thread successfully fills the order and decrements the inventory count by 1, setting the count to 0. The second thread detects that the inventory count is 0 and cancels the operation with a request to back-order the item.

Related Articles

  1. How to Check Object Type in Java
  2. How to Create a Jar File in Java
  3. How to Compile Packages in Java
  4. How to Throw an Exception in Java
  5. How to Create an Exception Class in Java
  6. How to Use the super Keyword to Call a Base Class Constructor in Java
  7. How to Use the Comparator.comparing Method in Java 8
  8. How to Use System.in in Java
  9. How to Call an Interface Method in Java
  10. How to Add a Time Zone in the Java 8 Date/Time API
  11. How to Rethrow an Exception in Java
  12. How to Use the instanceof Operator with a Generic Class in Java
  13. How to Instantiate an Object in Java
  14. How to Filter Distinct Elements from a Collection in Java 8
  15. How to Create a Derived Class in Java
  16. How to Skip Elements with the Skip Method in Java 8
  17. How to Create a Java Bean
  18. How to Implement an Interface in Java
  19. How to Compare Two Objects with the equals Method in Java
  20. How to Set PATH from JAVA_HOME
  21. How to Prevent Race Conditions in Java 8 (this article)
  22. How to Write a Block of Code in Java
  23. How to Display the Contents of a Directory in Java
  24. How to Group and Partition Collectors in Java 8
  25. How to Create a Reference to an Object in Java
  26. How to Reduce the Size of the Stream with the Limit Method in Java 8
  27. How to Write an Arithmetic Expression in Java
  28. How to Format Date and Time in the Java 8 Date/Time API
  29. How to Use Comparable and Comparator in Java
  30. How to Break a Loop in Java
  31. How to Use the this Keyword to Call Another Constructor in Java
  32. How to Write a Unit Test in Java
  33. How to Declare Variables in Java
  34. How to Override Base Class Methods with Derived Class Methods in Java
  35. How to Use Serialized Objects in Java
  36. How to Write Comments in Java
  37. How to Implement Functional Interfaces in Java 8
  38. How to Write Type Parameters with Multiple Bounds in Java
  39. How to Add Type and Repeating Annotations to Code in Java 8
  40. How to Use Basic Generics Syntax in Java
  41. How to Map Elements Using the Map Method in Java 8
  42. How to Work with Properties in Java
  43. How to Write while and do while Loops in Java
  44. How to Use the finally Block in Java
  45. How to Write for-each Loops in Java
  46. How to Create a Method in Java
  47. How to Continue a Loop in Java
  48. How to Handle Java Files with Streams
  49. How to Create an Interface Definition in Java
  50. How Default Base Class Constructors Are Used with Inheritance