对iOS锁的一些研究

#import <objc/runtime.h>
#import <objc/message.h>
#import <libkern/OSAtomic.h>
#import <pthread.h>
 
#define ITERATIONS (1024*1024*32)
- (void)testLock
{
    double then, now;
    unsigned int i, count;
    pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
    OSSpinLock spinlock = OS_SPINLOCK_INIT;
    
    @autoreleasepool {
        NSLock *lock = [NSLock new];
        then = CFAbsoluteTimeGetCurrent();
        for(i=0;i<ITERATIONS;++i)
        {
            [lock lock];
            [lock unlock];
        }
        now = CFAbsoluteTimeGetCurrent();
        printf("NSLock: %f sec\n", now-then);
        
 
        then = CFAbsoluteTimeGetCurrent();
        for(i=0;i<ITERATIONS;++i)
        {
            pthread_mutex_lock(&mutex);
            pthread_mutex_unlock(&mutex);
        }
        now = CFAbsoluteTimeGetCurrent();
        printf("pthread_mutex: %f sec\n", now-then);
        
        
        then = CFAbsoluteTimeGetCurrent();
        for(i=0;i<ITERATIONS;++i)
        {
            OSSpinLockLock(&spinlock);
            OSSpinLockUnlock(&spinlock);
        }
        now = CFAbsoluteTimeGetCurrent();
        printf("OSSpinlock: %f sec\n", now-then);
        
        id obj = [NSObject new];
        
        then = CFAbsoluteTimeGetCurrent();
        for(i=0;i<ITERATIONS;++i)
        {
            @synchronized(obj)
            {
            }
        }
        now = CFAbsoluteTimeGetCurrent();
        printf("@synchronized: %f sec\n", now-then);
    }
    
    
}

最近因为程序中频繁使用到了锁,不知道各种锁对性能的影响,今天稍作测试。顺便研究下里面的机制。

测试代码

测试原理:在一个线程中,对空代码段执行指定次数的加解锁。算出时间差。

测试代码请访问 Gist

测试结果

  • NSLock: 2.694002 sec
  • pthread_mutex: 1.761547 sec
  • OSSpinlock: 0.362136 sec
  • @synchronized: 6.849070 sec

可以看出来,用synchronized效率是最低的,而OSSpinlock效率高到无法直视。

详解synchronized OSSpinlock

@synchronized

创建给@synchronized指令的对象是一个用来区别保护块的唯一标示符。如果你在两个不同的线程里面执行上述方法,每次在一个线程传递了一个不同的对象给anObj参数,那么每次都将会拥有它的锁,并持续处理,中间不被其他线程阻塞。然而,如果你传递的是同一个对象,那么多个线程中的一个线程会首先获得该锁,而其他线程将会被阻塞直到第一个线程完成它的临界区。
作为一种预防措施,@synchronized块隐式的添加一个异常处理例程来保护代码。该处理例程会在异常抛出的时候自动的释放互斥锁。这意味着为了使用@synchronized指令,你必须在你的代码中启用异常处理。了如果你不想让隐式的异常处理例程带来额外的开销,你应该考虑使用锁的类。

OSSpinlock 官方描述

Spin locks are a simple, fast, thread-safe synchronization primitive that is suitable in situations
where contention is expected to be low. The spinlock operations use memory barriers to synchronize
access to shared memory protected by the lock. Preemption is possible while the lock is held.

结论

如果只是粗略的使用锁,不考虑性能问题可以使用synchronized。
如果对效率有较高的要求,还是采用OSSpinlock比较好。
因为Pthread的锁在也是用 OSSpinlock 实现的。OSSpinlock 的实现过程中,并没有进入系统kernel,使用OSSpinlock可以节省系统调用和上下文切换。


resources

@synchronized, NSLock, pthread, OSSpinLock showdown, done right

第四章 线程同步

Coneboy版权所有

转载注明出处

西安 iOS 开发

测试代码


已标记关键词 清除标记
相关推荐
【为什么还需要学习C++?】 你是否接触很多语言,但从来没有了解过编程语言的本质? 你是否想成为一名资深开发人员,想开发别人做不了的高性能程序? 你是否经常想要窥探大型企业级开发工程的思路,但苦于没有基础只能望洋兴叹?   那么C++就是你个人能力提升,职业之路进阶的不二之选。 【课程特色】 1.课程共19大章节,239课时内容,涵盖数据结构、函数、类、指针、标准库全部知识体系。 2.带你从知识与思想的层面从0构建C++知识框架,分析大型项目实践思路,为你打下坚实的基础。 3.李宁老师结合4大国外顶级C++著作的精华为大家推出的《征服C++11》课程。 【学完后我将达到什么水平?】 1.对C++的各个知识能够熟练配置、开发、部署; 2.吊打一切关于C++的笔试面试题; 3.面向物联网的“嵌入式”和面向大型化的“分布式”开发,掌握职业钥匙,把握行业先机。 【面向人群】 1.希望一站式快速入门的C++初学者; 2.希望快速学习 C++、掌握编程要义、修炼内功的开发者; 3.有志于挑战更高级的开发项目,成为资深开发的工程师。 【课程设计】 本课程包含3大模块 基础篇 本篇主要讲解c++的基础概念,包含数据类型、运算符等基本语法,数组、指针、字符串等基本词法,循环、函数、类等基本句法等。 进阶篇 本篇主要讲解编程中常用的一些技能,包含类的高级技术、类的继承、编译链接和命名空间等。 提升篇: 本篇可以帮助学员更加高效的进行c++开发,其中包含类型转换、文件操作、异常处理、代码重用等内容。
©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页