1. Lock classification

In OC, locks are divided into mutex and spin locks.

1. The spin lock

Mutex is a lock that protects a multithreaded shared resource. It differs from a mutex in that a spin lock iterates through a busy waiting loop to check whether the lock is available while trying to acquire it. When the previous thread’s task is not finished (locked), the next thread will wait (no sleep), and when the previous thread’s task is finished, the next thread will execute immediately.

In a multi-CPU environment, using a spin lock instead of a common mutex can often improve performance for programs with short locks.

Spin lock: OSSpinLock, read/write lock

2. The mutex

When a thread on the task is not completed (lock) when the next thread to sleep, waiting for the task has been completed, when a thread on the task has been completed under a thread will automatically wake up and then perform a task, the task will not immediately, but be executable status (ready). Mutex, used to ensure that only one thread can access the object at any time.

  • Mutex function

    Define a set of MUtex functions in Posix Threads specifically for Thread synchronization. Mutex, used to ensure that only one thread can access the object at any time. When the lock acquisition operation fails, the thread goes to sleep and wakes up waiting for the lock to be released.

    NSLock, NSCondtion, and NSRecursiveLock all encapsulate pThreads.

  • Understanding mutual exclusion and synchronization

    • Mutually exclusive: two threads processing, only one thread can run at a time;
    • Synchronization: In addition to the meaning of mutual exclusion, there are certain order requirements, that is, in accordance with a certain order.
  • Recursive locking

    The same thread can lock N times without causing a deadlock NSRecursiveLock, @synchronized, pthread_mutex(recursive)

Exclusive locks: Pthread_mutex, @synchronized, NSLock, NSCondition conditionlock, NSRecursiveLock, dispatch_se Maphore_t (semaphore)

3. Features of spin locks and mutex

  • Spin locks are busy, which means that when a locked resource is accessed, the caller thread does not sleep, but loops there until the locked resource releases the lock.

  • The mutex will sleep, which means that when accessing a locked resource, the caller thread will sleep. At this point, the CPU can schedule other threads to work until the locked resource releases the lock. The dormant thread is awakened.

  • Advantages and disadvantages of spinlocks

    The advantage is that because the spin lock does not cause the caller to sleep, there are no time-consuming operations such as thread scheduling, CPU time slice rotation, etc. So if the lock can be acquired in a very short time, spin locks are far more efficient than mutex locks.

    The downside is that the spin lock takes up CPU all the time. It’s running spin all the time without getting the lock, so it takes up CPU, and if you can’t get the lock in a very short time, it definitely makes the CPU less efficient. Spin locks do not make recursive calls.

4. Lock performance

The performance of the locks in the figure is in descending order: OSSpinLock -> dispatch_semaphone -> pthread_mutex -> NSLock -> NSCondition -> Pthread_mutex -> NSRecursiveLock -> NSConditionLock -> synchronized

2. Lock function

Let’s analyze it through a case. Simulate a ticketing process, with a total of 20 tickets and 4 Windows selling tickets at the same time, tracking the remaining tickets in real time. See the following code:

@interface ViewController () @property (nonatomic, assign) NSUInteger ticketCount; @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; self.ticketCount = 20; [self testSaleTicket]; } - (void)testSaleTicket{ dispatch_async(dispatch_get_global_queue(0, 0), ^{ for (int i = 0; i < 5; i++) { [self saleTicket]; }}); dispatch_async(dispatch_get_global_queue(0, 0), ^{ for (int i = 0; i < 5; i++) { [self saleTicket]; }}); dispatch_async(dispatch_get_global_queue(0, 0), ^{ for (int i = 0; i < 3; i++) { [self saleTicket]; }}); dispatch_async(dispatch_get_global_queue(0, 0), ^{ for (int i = 0; i < 10; i++) { [self saleTicket]; }}); } - (void)saleTicket{ if (self.ticketCount > 0) { self.ticketCount--; Sleep (0.1); NSLog(@" remaining tickets: %lu ",(unsigned long)self.ticketCount); } else {NSLog(@" current tickets sold out "); } } @endCopy the code

The running result is shown as follows:

Through the running results, it is found that due to the asynchronous operation, there are data insecurity problems and data chaos. We usually secure data by locking it so that only one thread can access the object at any one time.

Modify the case above the object as shown below:

Add a @synchronized mutex, rerunge the program, and find that it works and that the data is secure. @synchronized is more convenient, more readable, and the one we use most often.

3.@synchronized Implementation Principle

Given the above example, what does @synchronized do? That’s what we need to study and analyze.

1. Low-level exploration

  • Clang analyzes the implementation principle

    Provide the following code to see the underlying implementation principle through Clang, with the following figure:

    After clang, generate a.cpp file, punch the.cpp file, and locate it at the corresponding location of the main function. See below:

    As you can see, the objc_sync_enter method is called and a try-catch is used. In normal processing, the _SYNC_EXIT structure is provided and the corresponding destructor, objc_sync_exit, is also called.

  • Viewing assembly process

    First of all, we can analyze what’s going on at the bottom through assembly. Get the following information by setting breakpoints and turning on assembly debugging:

    From the assembly we can see that two underlying methods are called objc_sync_enter and objc_sync_exit, which are literally enter and exit, respectively. This is the same result as seen in Clang.

2. Implementation principle

The realization principle is analyzed in libobJC. dylib source code. Search the objc_sync_enter and objc_sync_exit methods source code implementation:

// enter int objc_sync_enter(id obj) { int result = OBJC_SYNC_SUCCESS; if (obj) { SyncData* data = id2data(obj, ACQUIRE); ASSERT(data); data->mutex.lock(); } else { // @synchronized(nil) does nothing if (DebugNilSync) { _objc_inform("NIL SYNC DEBUG: @synchronized(nil); set a breakpoint on objc_sync_nil to debug"); } objc_sync_nil(); } return result; } // exit int objc_sync_exit(id obj) { int result = OBJC_SYNC_SUCCESS; if (obj) { SyncData* data = id2data(obj, RELEASE); if (! data) { result = OBJC_SYNC_NOT_OWNING_THREAD_ERROR; } else { bool okay = data->mutex.tryUnlock(); if (! okay) { result = OBJC_SYNC_NOT_OWNING_THREAD_ERROR; } } } else { // @synchronized(nil) does nothing } return result; }Copy the code

Read the source code found that the implementation process of enter method and exit method is one-to-one correspondence.

Objc_sync_nil () is not found in libobjc.dylib. If obj is null, the lock is empty and nothing is done.

If obj is not empty, a SyncData object is wrapped in the Enter method and the mutex property is called by locking (); In the exit method, also get the corresponding SyncData object and call data-> mutex.tryunlock (); Unlock the account.

  • SyncData structure analysis

    The underlying definition of SyncData is as follows:

        typedef struct alignas(CacheLineSize) SyncData {
            struct SyncData* nextData;
            DisguisedPtr<objc_object> object;
            int32_t threadCount;  // number of THREADS using this block
            recursive_mutex_t mutex;
        } SyncData;
    
    Copy the code
    • struct SyncData* nextData;Contains an identical data structure, indicating that it is a single necklace table structure
    • objectuseDisguisedPtrPackaged
    • threadCountThe number of threads, how many threads lock the object
    • recursive_mutex_t mutex;Recursive locking

    Judging from the properties of SyncData, @synchronized supports recursive locking and supports multithreaded access.

  • StripedMap data structure

    The first step is to analyze the underlying data store structure. SyncData is stored in a hash table and is static. See the following code:

    static StripedMap<SyncList> sDataLists; class StripedMap { #if TARGET_OS_IPHONE && ! TARGET_OS_SIMULATOR enum { StripeCount = 8 }; #else enum { StripeCount = 64 }; #endif }Copy the code

    The table provides different capacities for different architectural environments, with capacities of 8 for the real environment and 64 for the simulated environment. And its element is SyncList, whose data structure is:

    struct SyncList {
        SyncData *data;
    
        spinlock_t lock;
    
        constexpr SyncList() : data(nil), lock(fork_unsafe_lock) { }
    };
    Copy the code

    SyncData is a linked list structure, thus forming a zipper structure. See below:

  • Id2data method

    Id2data implementation source code see the following figure:

    There are three major steps. First, through TLS, SyncData of the current thread is obtained from the thread cache for relevant processing. If the corresponding SyncData exists in the cache, it is fetched from the cache and processed; Finally, it includes some internal initialization of the insert cache and so on.

There are many branches of the process. Which processes will be invoked? The following analysis is carried out through cases combined with LLDB debugging.

3. Case tracking

Single thread recursive lock object invariant

Introduce the following example of adding the same lock recursively in a child thread. See below:

  • Breakpoint 1: Line 104 of the case

    Run the program, set a breakpoint at line 104 of the case, and trace into the ID2data method. All 64 data in the StripedMap table are empty. See below:

    To continue tracing, the tls_get_direct method is called to obtain the SyscData of the current thread binding. The data is null because it is the first time the lock is performed. See below:

    The corresponding SyncData is then fetched from the current thread’s cache list, which obviously does not store the object in the cache at this point, so it is also empty. See below:

    If SyncData bound by the current thread and SyncData in the cache list corresponding to the thread are empty, it will be fetched from the hash table, and there is no corresponding data in the current table, as shown in the following figure:

    The corresponding SyncData is not found in any of the above three places, and eventually a SyncData is created and the data is inserted into the corresponding LISTP header using header interpolation. See below:

    After SyncData is created, it is bound to the current thread (only one is bound to a thread and does not change after binding). Note that it is not saved to the cache list corresponding to the thread. See below:

    Return result to complete the lock function.

  • Breakpoint 2: Line 107 of the case

    From this breakpoint, the object is locked for a second time. Enter the id2data method, at this time there is already a data in the hash table, that is, the object corresponding to the listP is no longer empty (the same object). See below:

    Continue to run the program and get the SyncData bound by the current thread again, no longer null and with the same object. See below:

    The object corresponding to SyncData bound by the thread is the same as the object at this time. The lock is created again and the lock times are ++, as shown in the following figure:

  • Breakpoint 3: Line 110 of the case

    For the third time, since the object has not changed and the thread has not changed, the hash table is still one element, and the corresponding listp has only one element, the lock will change to 3. See below:

Single threaded recursive lockingobjectchange

To introduce the following example, we start our analysis directly from the second breakpoint, as shown below:

For the first breakpoint, which we have already analyzed, a new SyncData is created and bound to the current thread.

  • Breakpoint 2: Line 108 of the case

    Object is person2, and the thread has bound SyncData corresponding to Person1, so the thread binding is already occupied, but object is different. See below:

    Because the Person2 object is locked for the first time, there is no SyncData in the cache list or listp corresponding to the thread. See below:

    When person2 first comes in, it creates the object and puts SyncData into the cache list. See below:

    If person2 is locked again the next time, it will be retrieved from the cache list. If person1 is locked again, it will be fetched from the current thread, since the current thread is already bound to SyncData for Person1.

Multithreaded recursive lockingobjectchange

Introduce the following example, as shown below:

In the example above, the first two locking processes are not analyzed here, as is the case with single threads, so we start with multithreading, which is line 113.

  • Breakpoint 1: Line 113 of the case

    Tracing from breakpoint 1 into the id2Data method, the number of data in the hash table is 2, which is the two SyncData added by the outer thread. See below:

    As we continue to trace the code, we get the SyncData bound to the thread, which is NULL. Since the thread is new and has not locked, the binding data is NULL. FastCacheOccupied =NO See below:

    Then, the corresponding SyncData is retrieved from the cache list, which is also NULL, so the cache list here is also thread specific. See below:

    Next, the corresponding data is retrieved from the listp. In the outer thread, SyncData for Person1 and Person2 has been added, so it is available. If threadCount is increased by 1, the number of threads becomes 2**, as shown in the figure below:

    We will bind the SyncData to the current thread because fastCacheOccupied=NO; each thread will bind the first object by default.

  • Breakpoint 2: Line 116 of the case

    If person2 is locked, the SyncData bound by the current thread will be acquired first, because person1 has already been bound, and TLS corresponds to a different Object.

    It will fetch SyncData from the cache list corresponding to the thread, because the current thread has not added it, so it cannot be queried here, and will eventually fetch the corresponding SyncData from the listp. A threadCount plus 1 operation is performed at the same time. When this is done, the SyncData is added to the cache list corresponding to the thread. See below:

The logic of the flow in the new thread is the same as that in the outer thread, except that the thread-bound data and cache list data are different.

4.@synchronized Principle Summary

From the above analysis, objc_sync_Enter yields the following flow chart, where the property mutex.lock() is called after SyncData is acquired; Lock. See below:

Note: When a thread is opened, TLS and cache must be null. The cache must be searched in listP or create. The first Object added to a thread is bound to TLS and does not change in the current thread. If TLS is set up, any SyncData added later will be added to the cache list.

The objc_sync_exit process, in contrast, also calls the ID2Data method, gets SyncData, and decrement lockCount and threadCount. If count is equal to 0, it is removed from the corresponding binding relationship and cache list.

To sum up: @synchronized is a recursive lock that supports multiple threads.

4. Use of NSLock and NSRecursiveLock

To introduce an example, see the following code:

- (void)lg_testRecursive{ for (int i= 0; i<10; i++) { dispatch_async(dispatch_get_global_queue(0, 0), ^{ static void (^testMethod)(int); testMethod = ^(int value){ if (value > 0) { NSLog(@"current value = %d",value); testMethod(value - 1); }}; testMethod(10); }); }}Copy the code

In this case, 10 threads are opened for 10 to 1 output printing, and the above code is run. The results are shown in the following figure:

TestMethod is a static block, and the result is out of order due to multithreading. How to solve the problem of confusion? Lock! But where does the lock go? Obviously testMethod (10); Where the method is called is the core step. See below:

The verification results are consistent with our analysis. We know that the data structure SyncData contains recursive_mutex_t recursive mutex. @synchronized is a lock that supports multithreading recursion.

1.NSLock

Use NSLock to lock, add the lock in the outer layer of the business code, the modified code is as follows:

When NSLock is in effect, we usually place the lock inside the testMethod method during development, as shown in the following figure:

The program does not output normally. Why? TextMethod is called, lock is locked, and testMethod is called internally, causing the lock to be repeated.

hereNSLockRecursive locking is not supported!

2.NSRecursiveLock

Use NSRecursiveLock to add the lock to the outer layer of the business code and modify the code as follows:

It works as expected, but what if YOU add NSRecursiveLock inside your business code?

From the above results, NSRecursiveLock crashes after completing a business operation.

instructionsNSRecursiveLockRecursive locking within a single thread is supported, but multithreaded recursion is not supported.

5. The use of NSCondition

The object of NSCondition actually acts as a lock and a thread inspector: the lock is mainly used to protect the data source and perform the tasks triggered by the condition when the condition is detected; The thread checker mainly decides whether to continue running a thread based on conditions, that is, whether the thread is blocked.

  • Apis provided by NSCondition

    1. [condition lock];Generally, multiple threads access and modify the same data source at the same time to ensure that the data source can be accessed and modified only once at the same timelockOutside waiting, only tounlock, can be accessed
    2. [condition unlock];withlockAt the same time use
    3. [condition wait];Leave the current thread in wait state
    4. [condition signal]; CPUSignal the thread to stop waiting and continue executing
  • Case analysis

    Simulate the demand of production and consumption, open multiple threads for product production, open multiple threads for product sales at the same time. See the following examples:

    #pragma mark **-- NSCondition** - (void)lg_testConditon{ NSCondition *testCondition = [[NSCondition alloc] init]; // create a producer for (int I = 0; i < 50; i++) { dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{ [self lg_producer]; }); dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{ [self lg_producer]; }); } // create -consumer for (int I = 0; i < 50; i++) { dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{ [self lg_consumer]; }); dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{ [self lg_consumer]; }); } } - (void)lg_producer{ [_testCondition lock]; Self.ticketcount = self.ticketCount + 1; NSLog(@" produce an existing count %zd",self.ticketCount); [_testCondition signal]; // signal [_testCondition unlock]; } - (void) lg_consumer { [_testCondition lock]; If (self.ticketCount == 0) {NSLog(@" wait for count %zd",self.ticketCount); [_testCondition wait]; Self.ticketcount -= 1; self.ticketcount -= 1; NSLog(@" consume a remaining count %zd ",self.ticketCount); [_testCondition unlock]; }Copy the code

    Here we need to pay attention to the production line, consumption line need to lock processing, in order to ensure multithreading safety, at the same time, production and consumption before there is also a relationship, such as the number of inventory safety! Through [_testCondition wait]; Simulated inventory shortage, so that the consumption window to stop consumption; With [_testCondition signal]; Simulate that the existing inventory is ready to be consumed, and signal the waiting thread to start execution.

    See the following figure for the operating effect:

6.Foundation source code to understand lock packaging

NSLock, NSCondtion, and NSRecursiveLock encapsulate pThreads. Let’s explore the underlying implementation. Since lock-related content in the OC environment is encapsulated in the Fundation framework, the Fundation framework is not open source, so you can only view some statements:

In implementation, an NSLocking protocol is defined and lock and unlock methods are provided. So we use conditional locks, recursive locks that have lock methods and unlock methods.

The Fundation framework in Swift is open source. We can explore the implementation principle of locks through the Fundation framework in Swift environment.

  • NSLock

    Search NSLock: to obtain the NSLock lock definition.

    In the constructor init, the pthread_mutex_init function is called underneath. See below:

    The lock method calls the pthread_mutex_lock function; In the unlock method, the pthread_mutex_lock function is called. The bottom layer is the encapsulation of pThreads.

  • NSRecursiveLock

    In the same way, search for recursive locks NSRecursiveLock. See below:

    Found it is also rightpthreadEncapsulation, and through ininitMethod, throughpthread_mutexattr_settype(attrs, Int32(PTHREAD_MUTEX_RECURSIVE))Set it recursively.

  • NSCondition

    The bottom layer of NSCondition also encapsulates pthreads, as shown in the following figure:

    In addition to pthread_mutex, pthread_cond is processed and wait, signal, and BroadCase methods are provided.

  • NSConditionLock

    NSConditionLock does not operate pthread_mutex directly.

    However, the implementation provides an NSCondition attribute and a pthread_T attribute, through which locking and thread-related processing is implemented.

7. NSLocationLock analysis

NSConditionLock is also a conditional lock, and once one thread has acquired the lock, the other threads must wait.

  • The relevant API

    • [xxxx lock];saidxxxExpect to acquire the lock if no other thread acquires the lock (no need to judge internalconditionIf another thread has already acquired the lock (either conditionally or unconditionally), it will wait until the other thread has unlocked it.
    • [XXX lockWhenCondition:A condition];Indicates if no other thread has acquired the lock, but the internal lockconditionIs not equal toA condition, it still can’t get the lock, still waiting. If the internalconditionIs equal to theA condition, and no other thread to obtain the lock, then enter the code area, at the same time set it to obtain the lock, any other thread will wait for the completion of its code until it unlocked;
    • [XXX unlockWithCondition:A condition];Means to release the lock while putting the internalconditionSet toA condition;
    • Return = [XXX lockWhenCondition:A conditional beforeDate:A time];Indicates that the thread is no longer blocked if the lock is not acquired and the time elapsed. But note that the value returned isNOIt does not change the state of the lock. The purpose of this function is to handle both states.
  • Case analysis

    Introduce an example to understand NSLocationLock. See the following code:

    - (void)lg_testConditonLock{// Create conditional lock NSConditionLock *conditionLock = [[NSConditionLock alloc] initWithCondition:2]; dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{ [conditionLock lockWhenCondition:1]; NSLog (@ "thread 1"); [conditionLock unlockWithCondition:0]; }); dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0), ^{ [conditionLock lockWhenCondition:2]; Sleep (0.1); NSLog (@ "thread 2"); [conditionLock unlockWithCondition:1]; }); dispatch_async(dispatch_get_global_queue(0, 0), ^{ [conditionLock lock]; NSLog (@ "thread 3"); [conditionLock unlock]; }); }Copy the code

    What is the order of execution of the above cases? The running result is shown in the following figure:

    The key is what the lockWhenCondition method does! The following analysis:

    • NSConditionLockCreate condition when set2In other words, the conditions need to be met2Otherwise, it will not be executed.
    • Thread 1call[NSConditionLock lockWhenCondition:1], because the current condition is not met, it will enterwaitingState, currently entered towaiting, the current mutex is released.
    • Current at this timeThread 3call[NSConditionLock lock:], is essentially a call[NSConditionLock lockBeforeDate:]I don’t need to compare conditional values here, soThread 3Prints;
    • The followingThread 2perform[NSConditionLock lockWhenCondition:2], because the conditional value is satisfied, soThread 2Prints, and is called when it’s finished printing[NSConditionLock unlockWithCondition:1],At this time willvalueSet to1And sendboradcastAt this time,Thread 1Received the current signal, wake up to execute and print;
    • From the current print asThread 3->Thread 2->Thread 1;
    • [NSConditionLock lockWhenCondition:] :This is going to be based on incomingconditionValues andValueValues, and if they’re not equal, it blocks, goes into the thread pool, otherwise it continues;
    • [NSConditionLock unlockWithCondition:]:This will change the current one firstvalueValue, and then broadcast to wake up the current thread.

8. To read and write locks

  • Conceptual understanding

    Read/write lock is actually a special kind of spin lock. It divides the visitors to the shared resource into readers and writers. Readers only read the shared resource, while writers write the shared resource.

    This type of lock improves concurrency over a spin lock because in a multiprocessor system it allows multiple readers to access a shared resource at the same time, with the maximum possible number of readers being the actual number of logical cpus. Writers are exclusive; a read/write lock can have only one writer or more readers at a time (depending on the number of cpus), but not both readers and writers. Preemption also fails during read/write lock holding.

    If the read-write lock currently has no reader and no writer, then the writer can acquire the read-write lock immediately, otherwise it must spin there until there are no writers or readers. If the read-write lock has no writer, the reader can acquire the read-write lock immediately, otherwise the reader must spin there until the writer releases the read-write lock.

    Only one thread can hold the read-write lock in write mode at a time, but more than one thread can hold the read-write lock at the same time. Because of this feature, when a write/write lock is in the write/lock state, all threads attempting to lock the lock are blocked until the lock is unlocked. When a read-write lock is in read-lock state, all threads attempting to lock it in read mode gain access, but if a thread wishes to lock it in write mode, it must wait until all threads release the lock.

    Generally, when the read-write lock is in the read-mode lock state, if another thread tries to lock in write mode, the read-write lock will block the subsequent read-mode lock request. In this way, the read-mode lock will not be occupied for a long time, while the waiting write mode lock request will be blocked for a long time. Read/write locks are suitable for situations where data structures are read much more often than written. The read-write lock is also called shared-exclusive lock because it can be shared while the write lock means exclusive lock.

  • API

    • pthread_rwlock_t lock;/ / structure
    • pthread_rwlock_init(&lock, null);/ / initialization
    • pthread_rwlock_rdlock(&lock);/ / read lock
    • pthread_rwlock_tryrdlock(&lock);// Read attempts to lock
    • pthread_rwlock_wdlock(&lock);/ / write lock
    • pthread_rwlock_trywdlock(&lock);// Write attempts to lock
    • pthread_rwlock_unlock(&lock);/ / unlock
    • pthread_rwlock_destory(&lock);/ / destroy
  • The use of pthread_rwlock_t

    In this example, ten threads are started to read and write at the same time.

    • Can realize multi-read, multi-read is not mutually exclusive
    • Single write, read and write mutually exclusive
    • Write about a mutex

    See the following implementation:

    #import <Pthread.h> @interface ViewController () @property (nonatomic, assign) NSUInteger ticketCount; @property (nonatomic,assign) pthread_rwlock_t lock; @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; self.ticketCount = 0; [self rwTest]; } - (void)rwTest {// initialize pthread_rwlock_init(&_lock, NULL); // dispatch_queue_t queue = dispatch_get_global_queue(0, 0); For (int I = 0; i<10; i++) { dispatch_async(queue, ^{ [self read]; }); dispatch_async(queue, ^{ [self read]; }); // dispatch_async(queue, ^{[self write]; }); }} -(void)read{pthread_rwlock_rdlock(&_lock); sleep(1); NSLog (@ "read... %zd", self.ticketCount); / / unlock pthread_rwlock_unlock (& _lock); } // write -(void)write{// write and lock pthread_rwlock_wrlock(&_lock); sleep(1); NSLog (@ "write... %zd", ++self.ticketCount); / / unlock pthread_rwlock_unlock (& _lock); } @endCopy the code

    The running result is shown in the following figure:

    Through the above case can reflect the read and write lock can only have one writer at the same time, and can ensure that multiple read at the same time.

  • GCD fence function use

    The above requirements can also be met through the fence function.

    #import <pthread.h> @interface ViewController () @property (nonatomic, assign) NSUInteger ticketCount; @property (nonatomic, strong) dispatch_queue_t qCONCURRENT; @property (nonatomic, strong) dispatch_queue_t qSERIAL; @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; self.ticketCount = 0; Self. QCONCURRENT = dispatch_queue_create("selfCONCURRENT", DISPATCH_QUEUE_CONCURRENT); self.qSERIAL = dispatch_queue_create("selfSERIAL", DISPATCH_QUEUE_SERIAL); [self go_testReadWrite]; } - (void)touchesBegan:(NSSet< touch *> *)touches withEvent:(UIEvent *)event{ Dispatch_barrier_async (self.qconcurrent, ^{[self writeAction]; }); } #pragma read wirte - (void)go_testReadWrite{// For (int I = 0; i < 2000; i++) { dispatch_async(self.qCONCURRENT, ^{ [self readAction]; }); dispatch_async(self.qCONCURRENT, ^{ [self readAction]; }); dispatch_async(self.qCONCURRENT, ^{ [self readAction]; }); }} - (void)readAction {dispatch_async(self.qSERIAL, ^{sleep(1); NSLog (@ "read... %ld ------ %@", self.ticketCount, [NSThread currentThread]); }); } - (void)writeAction { sleep(1); NSLog (@ "write... %ld ------ %@", ++self.ticketCount, [NSThread currentThread]); }Copy the code

    The running result is shown in the following figure: