Changes for page Concurrency
Last modified by chrisby on 2024/06/02 15:15
Summary
-
Page properties (1 modified, 0 added, 0 removed)
Details
- Page properties
-
- Content
-
... ... @@ -1,11 +1,11 @@ 1 1 Objects are abstractions of processing, **threads are abstractions of timing**. 2 2 3 -### Why concurrency?3 +### Why Concurrency? 4 4 5 5 * **Concurrency is a decoupling strategy**. The what is decoupled from the when. 6 6 * **Concurrency is can improve the throughput and structure** of an application. 7 7 8 -### Why notconcurrency?8 +### Why Not Concurrency? 9 9 10 10 * **Unclean**: It is hard to write clean concurrent code, and it is harder to test and debug. 11 11 * **Design Changes**: Concurrency doesn't always improve performance behavior and but it always requires fundamental design changes. ... ... @@ -14,101 +14,32 @@ 14 14 * **Unreproducible**: Concurrency bugs are usually not reproducible; therefore, they are often written off as one-time occurrences (cosmic rays, glitches, etc.) rather than treated as true defects, as they should be. 15 15 * **Side-Effects**: When threads access out-of-sync data, incorrect results may be returned. 16 16 17 -### Principles ofDefensive Concurrency Programming17 +### Defensive Concurrency Programming 18 18 19 19 * **Single-Responsibility Principle** 20 20 * **Separation of code**: Changes to concurrent code should not be mixed with changes to the rest of the code. So you should separate the two cleanly. 21 21 * **Separation of change**: Concurrent code has special problems that are different, and often more serious, than sequential code. This means that concurrent and sequential code should be changed separately, not within the same commit, or even within the same branch. 22 22 * **Principle of Least Privilege**: Limit concurrent code to the resources it actually needs to avoid side effects. Minimize the amount of shared resources. Divide code blocks and resources into smaller blocks to apply more granular, and therefore more restrictive, resource access. 23 -* **Data Copies**: You can sometimes avoid shared resources by either working with copies of data and treating them as read-only objects, or by making multiple copies of data, having multiple threads compute results on them, and merging those results into a single thread. It is often worth creating multiple objects to avoid concurrency problems.23 +* **Data Copies**: You can sometimes avoid shared resources by either working with copies of data and treating them as read-only, or by making multiple copies of the data, having multiple threads compute results on them, and merging those results into a single thread. It is often worth creating multiple objects to avoid concurrency problems. 24 24 * **Independence**: Threads should be as independent as possible. Threads should not share their data or know anything about each other. Instead, they should prefer to work with their own local variables. Try to break data into independent subsets that can be processed by independent threads, possibly in different processes. 25 25 26 -### Thingsto learnbeforeworkingwith concurrency26 +### Basic Knowledge 27 27 28 -* **Get to know your library**: Use the thread-safe collections provided. Use non-blocking solutions if possible. Be aware multiple library classes are not thread safe. 29 -* Get to know execution models 30 - * Basic definitions 31 - * Bound Resources 32 - * Mutual Exclusion 33 - * Starvation 34 - * Deadlock 35 - * Livelock 36 - * Thread Pools 37 - * Future 38 - * Also these?? 39 - * Synchronization: General term for techniques that control the access of multiple threads to shared resources. 40 - * Race Condition: A situation where the system's behavior depends on the relative timing of events, often leading to bugs. 41 - * Semaphore: An abstract data type used to control access to a common resource by multiple threads. 42 - * Locks: Mechanisms to ensure that only one thread can access a resource at a time. 43 - * Atomic Operations: Operations that are completed in a single step relative to other threads. 44 - * Thread Safety 45 - * Race Conditions 46 - * Statelessness, Statefulness 47 - * Functional Programming 48 - * Cloning Data to avoid side effects 49 - * Side effects 50 - * Producer-consumer 51 - * Reader Writer 52 - * Philosopher problem → Study algorithms and their application in solutions. 28 +Before starting to write concurrent code, get familiar with the following basics: 53 53 54 -A few more suggestions from ChatGPT: 30 +* **Libraries**: Use the thread-safe collections provided. Use non-blocking solutions if possible. Be aware multiple library classes are not thread safe. 31 +* **Concepts**: Mutual Exclusion, Deadlock, Livelock, Thread Pools, Semaphore, Locks, Race Condition, Starvation, 32 +* **Patterns**: Producer-consumer, Reader-Writer 33 +* **Algorithms**: Study common algorithms and their use in solutions. For example, the Dining Philosophers problem. 55 55 56 - Learning about concurrency algorithms and common concurrency problems is a great way to deepen your understanding of concurrent programming. Here's a list of key algorithms and problems, along with their typical solutions: 57 - Concurrency Algorithms: 58 - 59 - Producer-Consumer: 60 - Problem: How to handle scenarios where one or more threads (producers) are producing data and one or more threads (consumers) are consuming it. 61 - Solution: Use buffers, queues, semaphores, or condition variables to synchronize producers and consumers. 62 - 63 - Readers-Writers: 64 - Problem: How to manage access to a shared resource where some threads (readers) only read data, and others (writers) write data. 65 - Solution: Implement mechanisms to ensure that multiple readers can access the resource simultaneously, but writers have exclusive access. 66 - 67 - Dining Philosophers: 68 - Problem: A classic synchronization problem dealing with resource allocation and avoiding deadlocks. 69 - Solution: Strategies include resource hierarchy, arbitrator, or limiting the number of philosophers. 70 - 71 - Barriers: 72 - Problem: Synchronizing a group of threads to wait until they have all reached a certain point in their execution. 73 - Solution: Use barrier constructs that block threads until all have reached the barrier. 74 - 75 - Concurrency Problems and Solutions: 76 - 77 - Deadlocks: 78 - Problem: Occurs when multiple threads or processes are waiting on each other to release resources, and none of them can proceed. 79 - Solution: Deadlock prevention techniques (like resource ordering), deadlock avoidance (like Banker’s algorithm), and deadlock detection and recovery. 80 - 81 - Race Conditions: 82 - Problem: Occurs when the outcome of a program depends on the relative timing of threads or processes. 83 - Solution: Use mutual exclusion (mutexes), atomic operations, or transactional memory to ensure that only one thread can access the shared resource at a time. 84 - 85 - Livelocks: 86 - Problem: Threads or processes are actively performing concurrent operations, but these operations do not progress the state of the program. 87 - Solution: Careful algorithm design to ensure progress and avoid situations where processes continuously yield to each other. 88 - 89 - Starvation: 90 - Problem: A thread or process does not get the necessary resources to proceed, while others continue to be serviced. 91 - Solution: Implement fair locking mechanisms, priority scheduling, or resource allocation strategies that ensure all processes get a chance to proceed. 92 - 93 - Priority Inversion: 94 - Problem: A lower-priority thread holds a resource needed by a higher-priority thread, leading to the higher-priority thread waiting unexpectedly. 95 - Solution: Priority inheritance protocols where the lower-priority thread temporarily inherits the higher priority. 96 - 97 - Thread Interference: 98 - Problem: When multiple threads are accessing and modifying shared data, causing unexpected results. 99 - Solution: Ensure that critical sections of code that access shared resources are protected using synchronization mechanisms like locks. 100 - 101 101 ### Watch out for dependencies between synchronized methods 102 102 103 -* Dependencies between synchronized methods in concurrent code cause subtle bugs. 104 -* Avoid applying more than one method to a shared object. If this is not possible, you have three options: 105 - * Client-based locking: the client should lock the server before the first method is called and ensure that the lock includes the code that calls the last method. 106 - * Server-based locking: Create a method in the server that locks the server, calls all methods, and then unlocks the server. Have the client call the new method. 107 - * Adapted Server: Create an intermediate component that performs the lock. This is a variant of server-based locking if the original server cannot be changed. 108 -* Keep synchronized sections small. 109 - * Locks are expensive because they add administrative overhead to delays. On the other hand, critical sections must be protected. 110 - * Critical sections, are parts of the code that are only executed correctly if several threads do not access it at the same time. 111 - * Keep synchronized sections as small as possible. 37 +* **Avoid dependencies between synchronized methods**: Synchronized means that only one thread can access a method at a time. In concurrent code, such dependencies, such as when one synchronized method calls another, can cause subtle bugs like deadlocks and performance issues. 38 +* **Avoid applying more than one method to a shared object.** If this is not possible, you have three options: 39 + * **Client-based locking**: The client locks the server, calls all the server methods, and then releases the lock. 40 + * **Server-based locking**: Create a method in the server that locks the server, calls all the methods, and then unlocks the server. A client can now safely call this new method. 41 + * **Adapted Server**: Create an intermediate component to perform the lock. This is a variant of server-based locking when the original server cannot be changed. 42 +* **Keep synchronized sections small. **Locks are expensive because they add administrative overhead and delay. On the other hand, critical sections need to be protected. Critical sections are pieces of code that will only run correctly if they are not accessed by multiple threads at the same time. Keeping synchronized sections small avoids both problems. 112 112 113 113 ### Writing correct shutdown code is difficult 114 114