Download presentation
Presentation is loading. Please wait.
Published byAugusta Byrd Modified over 8 years ago
1
Concurrent Programming Acknowledgements: Some slides adapted from David Evans, U. Virginia
2
2 Why Concurrent Programming Abstraction: Some problems are clearer to program concurrently – Modularity: Don’t have to explicitly interleave code for different abstractions (especially: user interfaces) – Modeling: Closer map to real world problems: things in the real world aren’t sequential Efficiency: Can leverage the power of multicores in modern machines
3
Learning Objectives Define threads and how to create them in Java Identify the challenges with writing multi- threaded code Apply synchronized methods to ADTs to avoid race conditions Use fine-grained locks in place of coarse grained locks Avoid locks entirely, if possible 3
4
Concept of threads A thread is an independent unit of control that lives in the same address space as the application, but has its own execution context Each thread has its own stack/local variables All objects on the heap are freely accessible to every thread in the program – shared data 4
5
Threads in Java Every thread in Java inherits from Thread class Thread’s constructor takes an object that implements an interface Runnable – Any class that implements Runnable, must implement a run() method with no arguments and no return value – The run method can do pretty much anything it wants to, including spawn new threads. 5
6
Using threads – Example 1 6 public class MyRunnable implements Runnable { private long start, end, sum; MyRunnable(long start, long end) { this.start = start; this.end = end; } public void run() { sum = 0; for (long i = start; i < end; i++) sum += i; } public long getSum() { return sum; } }
7
Using threads – Example 2 7 public static void main(String[] args) { Vector threads = new Vector (); Vector tasks = new Vector (); long sum = 0; for (int i = 0; i < 100; i++) { Runnable task = new MyRunnable(i * 10000, (i + 1) * 10000) ); Thread worker = new Thread(task); threads.add(worker); tasks.add(task). worker.start(); } sleep(100);// Wait for 100 ms (not required) for (int i = 0; i < threads.size(); ++i) { threads[i].join(); sum += tasks[j].getSum(); }
8
Learning Objectives Define threads and how to create them in Java Identify the challenges with writing multi- threaded code Apply synchronized methods to ADTs to avoid race conditions Use fine-grained locks in place of coarse grained locks Avoid locks entirely, if possible 8
9
Challenge of Concurrency DataStore Instruction Streams Coordinated access to shared data! Thread 1 Thread 2 Thread 3
10
10 Example: Scheduling Meetings Alice wants to schedule a meeting with Bob and Colleen BobAliceColleen “When can you meet Friday?” “11am or 3pm” “9am or 11am” “Let’s meet at 11am” Reserves 11am for meeting Reserves 11am for meeting Picks meeting time
11
11 Race Condition BobAliceColleen “When can you meet Friday?” “9, 11am or 3pm” “9am or 11am” “Let’s meet at 11am” Picks meeting time Doug “When can you meet Friday?” “9, 11am or 3pm” “Let’s meet at 11am” Reserves 11am for Doug “I’m busy then…”
12
12 Locking BobAliceColleen “When can you meet Friday?” “9, 11am or 3pm” “9am or 11am” “Let’s meet at 11am” Picks meeting time Doug “When can you meet Friday?” “3pm” “Let’s meet at 3” Locks calendar
13
13 Deadlocks BobAliceColleen “When can you meet Friday?” “9, 11am or 3pm” Doug “When can you meet Friday?” Locks calendar for Alice, can’t respond to Doug “When can you meet Friday?” Locks calendar for Doug, can’t respond to Alice Can’t schedule meeting, no response from Bob Can’t schedule meeting, no response from Colleen
14
Why are threads hard? Too few ordering constraints: race conditions Too many ordering constraints: deadlocks, poor performance, livelocks, starvation Hard/impossible to reason about modularly – If an object is accessible to multiple threads, need to think about what any of those threads could do at any time! Testing is even more difficult than for sequential code – Even if you test all the inputs, you don’t know it will work if threads run in different order
15
Learning Objectives Define threads and how to create them in Java Identify the challenges with writing multi- threaded code Apply synchronized methods to ADTs to avoid race conditions Use fine-grained locks in place of coarse grained locks Avoid locks entirely, if possible 15
16
Example: IntSet ADT public void insert (int x) { // MODIFIES: this // EFFECTS: adds x to the set such that // this_post = this u {x} if ( getIndex(x) < 0 ) elems.add( new Integer(x) ); } 16
17
Example: IntSet Thread 1 public void insert (int x) { // MODIFIES: this // EFFECTS: adds x to the set // this_post = this u {x} if ( getIndex(x) < 0 ) elems.add( new Integer(x) ); } Thread 2 public void insert (int x) { // MODIFIES: this // EFFECTS: adds x to the set // this_post = this u {x} if ( getIndex(x) < 0 ) elems.add( new Integer(x) ); } 17 Can you violate the rep invariant in any way ?
18
What’s the Problem ? If two threads execute the method simultaneously, it is possible for the rep invariant to be violated (i.e., no duplicates) – Occurs in some interleavings, but not others – Frustrating to track down problem – Leads to subtle errors in other parts of the IntSet We need to prevent multiple threads from executing the insert method simultaneously 18
19
Synchronized Methods In Java, you can declare a method as follows: public synchronized void insert( int x ) { … } Only one thread may execute a synchronized method at any point in time – Other threads queued up at the method’s entry – Enter the method when the current thread leaves the method or throws an exception 19
20
What happens in this case ? Thread 1 public synchronized void insert (int x) { // MODIFIES: this // EFFECTS: adds x to the set // this_post = this u {x} if ( getIndex(x) < 0 ) elems.add( new Integer(x) ); } Thread 2 pubic void remove(int x) { // MODIFIES: this // EFFECTS: this_post = this - {x} int i = getIndex(x); if (i < 0) return; // Not found elems.set(i, elems.lastElement() ); elems.remove(elems.size() – 1); } 20
21
What’s the problem ? The two methods conflict with each other – While a thread is inserting an object, another thread can remove the same object (so the post- condition will not be satisfied) – While a thread is removing an object, if a thread tries to insert a (different) object, the inserted object will be removed, and not the original one To solve the problem, the insert and remove methods should not execute simultaneously 21
22
synchronized to the rescue Luckily for us, Java allows multiple methods to be declared exclusive using the synchronized keyword public synchronized void insert(int x) { … }; public synchronized void remove(int x) { … }; Only one thread can execute any synchronized method of the object at any time Other threads queued up at the entry of their respective methods, and executed when this leaves 22
23
What about these methods ? public int size() { return elems.size(); } public boolean isIn(int x) { return getIndex(x) >= 0; } Constructors cannot be synchronized in Java 23
24
Synchronized Methods Declare two methods synchronized if and only if their simultaneous executions are incompatible – typically mutator methods Observers need not be declared synchronized – Provided they do not change the rep in any way – But you do not get any guarantees about the state when they are executed simultaneously with mutator methods – may result in inconsistencies 24
25
IntSet toString() method public String toString() { String str = “{ ”; for (i=0; i<elems.size(); ++i) { str = str + “ “ + elems.get(i).toString(); return str + “ }”; } Does this method need to be synchronized ? 25
26
Should toString be synchronized ? Consider the following code operating on an IntSet s = {2, 3, 5}. System.out.println( s.toString() ); Assume that another thread is executing this at the same time as the set is being printed: s.remove(3); This method can print {2, 3, 3} Breaks the RI 26
27
Take-aways Rules for synchronizing ADTs – Mutator methods of an ADT should almost always be synchronized – otherwise, unintended changes – Observer methods need not be synchronized only if they are simple return operations – Observer methods that access the ADTs data multiple times should be synchronized – Constructors should never be synchronized – Underlying data structures should be thread-safe 27
28
Group Activity Consider the intStack ADT which you had worked on previously. Which methods will you make synchronized and why ? You must make the minimal set of methods synchronized to ensure that the ADT’s implementation is correct What if your ADT had a peek method ? How would your answer change, if at all ? 28
29
Learning Objectives Define threads and how to create them in Java Identify the challenges with writing multi- threaded code Apply synchronized methods to ADTs to avoid race conditions Use fine-grained locks in place of coarse grained locks Avoid locks entirely, if possible 29
30
Granularity of Locks Coarse-grained Entire object is protected with a single lock Only one method may be executed at any time Easier to reason about correctness, but slow ! Fine-grained Each part of the object is protected with separate locks or not at all protected Allows multiple methods to be executed simultaneously Harder to reason about correctness, but faster ! 30
31
Example: hashTable Consider a hashtable where each table entry is stored in a vector. We can allow updates of one entry even while another entry is read, as the two do not interfere with each other. However, implementing this ADT using synchronized methods in Java will mean that all threads are locked out of the table when one is executing a synchronized method 31
32
Solution: Fine-grained locks Create lock objects and synchronize on them explicitly arbitrary code inside blocks synchronized(lock1) { doOperation1(); doOperation2(); } 32
33
Fine-grained locks: How to use them ? You will need as many lock objects as the number of synchronization operations needed in the ADT. For the hash-table example, you can update a hash-table entry as follows: int index = hashMap.get(key); synchronized( lock[index] ) { hashMap.update(key, value); } 33
34
Learning Objectives Define threads and how to create them in Java Identify the challenges with writing multi- threaded code Apply synchronized methods to ADTs to avoid race conditions Use fine-grained locks in place of coarse grained locks Avoid locks entirely, if possible 34
35
Locks are problematic … Cause unnecessary synchronization – Performance overheads can be extremely high May lead to deadlocks and/or starvation Not straightforward to compose code together which has locks can lead to deadlocks 35
36
Better to avoid locks entirely … But how ? – We care about the invariant/spec being preserved. Locking is only a means to the end. One solution: Use only immutable ADTs – Immutable ADTs cannot be modified, so no problem of concurrent modifications – Default in functional languages such as Haskell 36
37
Immutable ADTs Consider an ADT ImmutableIntSet, which does not allow insert/remove operations. The only operations it supports are union, and different both of which result in a new ImmutableIntSet Show that the ADT does not require any synchronized operations for correctness. 37
38
ImmutableIntSet ADT class ImmutableIntSet { private Vector elems; ImmutableIntSet(Vector e); public ImmutableIntSet union(ImmutableIntSet, ImmutableIntSet); public ImmutableIntSet difference (ImmutableIntSet, ImmutableIntSet); public int size(); public boolean isIn(int x); } 38
39
Group Activity Implement the union and difference operations of ImmutableIntSet, and show that they will be correct in a multi-threaded context even if no lock is taken during their execution. 39
40
Learning Objectives Define threads and how to create them in Java Identify the challenges with writing multi- threaded code Apply synchronized methods to ADTs to avoid race conditions Use fine-grained locks in place of coarse grained locks Avoid locks entirely, if possible 40
Similar presentations
© 2025 SlidePlayer.com. Inc.
All rights reserved.