“This is the 13th day of my participation in the Gwen Challenge in November. Check out the details: The Last Gwen Challenge in 2021”

Multithreaded competition

The phenomenon of description

In a real scenario, a resource may be shared by multiple threads, that is, multiple threads may access the same resource, object, and variable, which may lead to data corruption and data security issues (as shown in the following figure).

Example:

- (IBAction)threadLock:(UIButton *)sender {
    self.ticketCount = 5;
    
    self.thread01 = [[NSThread alloc]initWithTarget:self selector:@selector(saleTicket) object:nil];
    self.thread01.name = @"Conductor 1";
    self.thread02 = [[NSThread alloc]initWithTarget:self selector:@selector(saleTicket) object:nil];
    self.thread02.name = @"Conductor 2";
    self.thread03 = [[NSThread alloc]initWithTarget:self selector:@selector(saleTicket) object:nil];
    self.thread03.name = @"Conductor 3";
    
    [self.thread01 start];
    [self.thread02 start];
    [self.thread03 start];
}

- (void)saleTicket{
    while (1) {
        NSLog(@"To sell tickets -%@",[NSThread currentThread].name);
        NSInteger count = self.ticketCount;

        if (count > 0 ){
            self.ticketCount = count - 1;
            NSLog(@"%@ sold one ticket, %ld left",[NSThread currentThread].name,(long)self.ticketCount);
        }else{
            NSLog(@"Sold out.");
            break; }}}Copy the code

log:

Problem solving

In view of the above situation, through the thread lock to solve the above problem. Before thread A accesses the data source, it locks the data source, reads the data, calculates the data, writes the result to the data source, and unlocks it. Before thread B attempts to access the data source again, it locks the data source again, reads the data, performs calculations, and unlocks the data source after writing the results to it. This avoids data errors caused by multiple threads accessing the data at the same time (figure below)

Example:

- (void)saleTicket{

    while (1) {
        NSLog(@"To sell tickets -%@",[NSThread currentThread].name);
        // @synchronized, which must be a synchronized object, usually self
        @synchronized (self) {
        
            NSInteger count = self.ticketCount;
            if (count > 0 ){
                self.ticketCount = count - 1;
                NSLog(@"%@ sold one ticket, %ld left",[NSThread currentThread].name,(long)self.ticketCount);
            }else{
                NSLog(@"Sold out.");
                break; }}}}Copy the code

The log: