多用派发队列,少用同步锁
2016-07-29 09:01
309 查看
在Objective-C中,如果有多个线程要执行同一份代码,那么有时候可能会出现问题,这种情况下,通常使用锁来实现某种同步机制。在GCD出现之前,有两种办法。第一种采用内置的“同步块”(synchronization block):
- (void)synchronizedMethod { @synchronized(self){ // Safe } }
这种写法会根据给定的对象,自动创建一个锁,并等待块中的代码执行完毕。执行到这段代码的结尾处,锁就释放了。but滥用@synchronized(self)则会降低代码效率,因为共用同一个锁的那些同步块,都必须按顺序执行。
另一个办法是直接食用NSLock对象:
_lock = [[NSLock alloc] init]; -(void)synchronizedMethod { [_lock lock]; // Safe [_lock unlock]; }
GCD以更简单、更高效的形式为代码加锁。 使用“串行同步队列”。将读取操作以及写入操作都安排在同一个队列里,即可保证数据同步。
_syncQueue = dispatch_queue_create("com.effectiveobjectivec.syncQueue",NULL); - (NSString *)someString { __block NSString *localSomething; dispatch_sync(_syncQueue, ^{ localSomeString = _someString; }); return localSomeString; } - (void)setSomeString:(NSString *)someString { dispatch_sync(_syncQueue, ^{ _someString = someString; }); }
多个获取方法可以并发执行,而获取方法与设置方法之间不能并发执行,利用这个特点,可以优化代码。改为并发队列。
_syncQueue = dispatch_get_global_queue (DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); -(NSString *)someString { __block NSString *localSomeString; dispatch_sync (_syncQueue, ^{ localSomeString = _someString; }); return localSomeString; } - (void)setSomeString:(NSString *)someString { dispatch_barrier_async(_syncQueue, ^{ _someString = someString; }); }
栏栅块必须单独执行,不能与其他块并行,这只对并发队列有意义,因为串行队列中的块总是按顺序逐个来执行的。并发队列如果发现接下来要处理的块是个栏栅块,那么就一直等到当前所有并发块都执行完毕,才会单独执行这个栏栅块。待栏栅块执行过后,再按正常方式继续向下处理。
设置方法中使用了栏栅块之后,对属性的读取操作依然可以并发执行,但是写入操作却必须单独执行了。
相关文章推荐
- OC高效率52之多用派发队列,少用同步锁
- GCD之同步锁和派发队列
- 第41条:多用派发队列,少用同步锁
- 多用派发队列,少用同步锁
- linux同步(二)---等待队列
- GCD 主队列/并行队列/全局队列/主队列/串行队列/同步任务/异步任务区别
- FIFO队列并发同步访问 C++ 实现 ~~~
- iOS网络开发中的同步、异步和请求队列
- 【死磕Java并发】—– J.U.C之AQS:CLH同步队列
- 多ajax请求的各类解决方案(同步, 队列, cancel请求)
- 同步容器、并发容器、阻塞队列、双端队列
- No.7_7 OpenCL 同步——命令队列
- python多线程编程(6): 队列同步
- 多线程环境下不安全的消息队列存取---线程不同步会造成隐患
- iOS网络编程之同步、异步、请求队列
- 利用操作队列和信号量来实现操作同步
- FreeRTOS学习笔记——任务间使用队列同步数据
- Python线程编程—同步队列
- 使用Spring的FactoryBean创建同步队列线程池
- iOS网络开发中的同步、异步和请求队列