Threads are lightweight processes that share certain areas of the memory, unlike regular processes. This makes threads very efficient, but it also introduces additional complexities regarding memory management. You can protect your shared data by making it accessible (for reading and writing) to only one thread at a point of time. Other techniques include defining immutable classes (the states of which can’t be modified and defining volatile variables.
Identifying shared data:
A JVM instance hosts only one Java application. A Java application can create and execute multiple threads. When a new thread is created, it’s allotted its exclusive share in the memory. The JVM also allows partial runtime data to be shared between threads. A JVM’s runtime data set includes the method area, the heap, Java stacks, and PC registers (it also includes native method stacks, which aren’t covered by the exam, so they aren’t discussed here). The method area includes class information that includes a note on all its variables and methods. This data set includes the static variables, which are accessible and shared by all the threads in a JVM. The static vari- ables are shared by one class loader—that is, each class loader has its own set of static variables. The heap includes objects that are created during the lifetime of an application, again shared by multiple threads and processes. PC registers and Java stacks aren’t shared across the threads. On instantiation, each thread gets a PC regis- ter, which defines the next set of instructions to execute. Each thread is also assigned a Java stack. When a thread begins execution of a method, a method frame is created and inserted into the java stack, which is used to store the local variables, method parameters, return values, and intermediate values that might be used in a method.
Thread interference:
Interleaving of multiple threads that manipulate shared data using multiple steps leads to thread interference. You can’t assume that a single statement of Java code executes atomically as a single step by the processor. For example, a simple statement like incrementing a variable value might involve multiple steps like loading of the variable value from memory to registers (working space), incrementing the value, and reloading the new value in the memory. When multiple threads execute this seemingly atomic statement, they might interleave, resulting in incorrect variable values.
Operations that use arithmetic and assignment operators like ++, --, +=, -=, *=, and /= aren’t atomic. Multiple threads that manipulate variable values using these operators can interleave.
Let’s work with an example of a class Book, which defines an instance variable copies- Sold that can be manipulated using its methods newSale() or returnBook():
Let’s see what happens when another class, say, ShoppingCart, instantiates a Book and passes it to the threads OnlineBuy and OnlineReturn:
In the preceding code method main() starts three threads—task1, task2, and task3. These threads manipulate and share the same Book instance, book. The threads task1 and task2 execute book.newSale(), and task3 executes book.returnBook(). As mentioned previously, ++copiesSold and --copiesSold aren’t atomic operations. Also, as a programmer you can’t determine or command the exact time when these threads will start with their execution (it depends on how they’re scheduled to exe- cute by the OS). Let’s assume that task2 starts with its execution and reads book .copiesSold. Before it can modify this shared value, it’s also read by threads task3 and task2, which are unaware that this value is being modified by another thread. All these threads modify the value of book.copiesSold and update it back in order. The last thread to update the value book.copiesSold overrides updates of the other two threads. Figure below shows one of the possible ways in which the threads task1, task2, and task3 can interleave.
So, how can you assure that when multiple threads update shared values it doesn’t lead to incorrect results? How can you communicate between threads? Let’s discuss this in the next section.
Examples:
1)
importjava.util.Set;publicclassThread1 {publicstaticvoidmain(String[] args) {StringBuilder string =newStringBuilder("shared ");Thread thread =newProducerThread(string);Thread thread2 =newConsumerThread(string);thread.start();thread2.start();Set<Thread> seThreads =Thread.getAllStackTraces().keySet();seThreads.stream().map(Thread::getName).forEach(System.out::println);try {thread.join();thread2.join(); } catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace(); }System.out.println(string); }}classProducerThreadextendsThread {privateStringBuilder string =null;publicProducerThread(StringBuilder string) {this.string= string; } @Overridepublicvoidrun() {System.out.println("I am prod thread");int i =0;while (i <100) { i++;string.append("prod "+ i); } }}classConsumerThreadextendsThread {privateStringBuilder string =null;publicConsumerThread(StringBuilder string) {this.string= string; } @Overridepublicvoidrun() {System.out.println("I am con thread");int i =0;while (i <100) { i++;string.append("cons "+ i); } }}