当前位置: 移动技术网 > IT编程>移动开发>IOS > 谈谈iOS中的几种锁

谈谈iOS中的几种锁

2019年07月24日  | 移动技术网IT编程  | 我要评论

宗玉佩,搜房网666装修,中式吊灯

1 前言

近日工作不是太忙,刚好有时间了解一些其他东西,本来打算今天上午去体检,但是看看天气还是明天再去吧,也有很大一个原因:就是周六没有预约上!闲话少说,这里简单对锁来个简单介绍分享。

2 目录

  1. 第一部分:什么是锁
  2. 第二部分:锁的分类
  3. 第三部分:锁的作用
  4. 第四部分:ios中锁的实现

第一部分:什么是锁

从小就知道锁,就是家里门上的那个锁,用来防止盗窃的锁。它还有钥匙,用于开锁。不过这里的锁,并不是小时候认知的锁,而是站在程序员的角度的锁。这里我就按照我的理解来介绍一下锁。

在计算机科学中,锁是一种同步机制,用于在存在多线程的环境中实施对资源的访问限制。你可以理解成它用于排除并发的一种策略。看例子

if (lock == 0) {
lock = mypid;
}

上面这段代码并不能保证这个任务有个锁,因此它可以在同一时间被多个任务执行。这个时候就有可能多个任务都检测到lock是空闲的,因此两个或者多个任务都将尝试设置lock,而不知道其他的任务也在尝试设置lock。这个时候就会出问题了。
再看看这段代码:

class acccount {
long val = 0; //这里不可在其他方法修改,只能通过add/minus修改
object thislock = new object();
public void add(const long x) {
lock(thislock) {
  val +=x;
}
}
public void minus(const long x) {
lock(thislock) {
  val -=x;
  }
}
}

这样就能防止多个任务去修改val了,(这里注意,如果val是public的,那个也会导致一些问题)。

第二部分:锁的分类

锁根据不同的性质可以分成不同的类。

在wikipedia介绍中,一般的锁都是建议锁,也就四每个任务去访问公共资源的时候,都需要取得锁的资讯,再根据锁资讯来确定是否可以存取。若存取对应资讯,锁的状态会改变为锁定,因此其他线程不会访问该资源,当结束访问时,锁会释放,允许其他任务访问。有些系统有强制锁,若未经授权的锁访问锁定的资料,在访问时就会产生异常。
在ios中,锁分为递归锁、条件锁、分布式锁、一般锁(这里是看着nslock类里面的分类划分的)。
对于数据库的锁分类:

分类方式 分类
按锁的粒度划分 表级锁、行级锁、页级锁
按锁的级别划分 共享锁、排他锁
按加锁方式划分 自动锁、显示锁
按锁的使用方式划分 乐观锁、悲观锁
按操作划分 dml锁、ddl锁

这里就不在详细介绍了,感兴趣的大家可以自己查阅相关资料。

第三部分:锁的作用

这个比较通俗来讲:就是为了防止在多线程(多任务)的情况下对共享资源(临界资源)的脏读或者脏写。也可以理解为:执行多线程时用于强行限制资源访问的同步机制,即并发控制中保证互斥的要求。

第四部分:ios中锁的实现

先看看ios中nslock类的.h文件。这里就不在写上来了。从代码中可以看出,该类分成了几个子类:nslock、nsconditionlock、nsrecursivelock以及nscondition。然后有一个nslocking的协议:

@protocol nslocking
- (void)lock;
- (void)unlock;
@end

这几个子类都遵循了nslock的协议,这里简单介绍一下其中的几个方法:

对于trylock方法,尝试获取一个锁,并且立刻返回bool值,yes表示获取了锁,no表示没有获取锁失败。 lockbeforedate:方法,在某个时刻之前获取锁,如果获取成功,则返回yes,no表示获取锁失败。接下来就让我们看一下ios中实现锁的方式:

方式1 使用nslock类

- (void)nslockdemo {
  nslock *mylock = [[nslock alloc] init];
  _testlock = [[testlock alloc] init];
  dispatch_async(dispatch_get_global_queue(dispatch_queue_priority_default, 0), ^{
    [mylock lock];
    [_testlock method1];
    sleep(5);
    [mylock unlock];
    if ([mylock trylock]) {
      nslog(@"可以获得锁");
    }else {
      nslog(@"不可以获得所");
    }
  });
  dispatch_async(dispatch_get_global_queue(dispatch_queue_priority_default, 0), ^{
    sleep(1);
    if ([mylock trylock]) {
      nslog(@"---可以获得锁");
    }else {
      nslog(@"----不可以获得所");
    }
    [mylock lock];
    [_testlock method2];
    [mylock unlock];
  });
}

方式2 使用@synchorize

- (void)synchronizedemo {
  _testlock = [[testlock alloc] init];
  dispatch_async(dispatch_get_global_queue(dispatch_queue_priority_default, 0), ^{
    @synchronized (_testlock) {
      [_testlock method1];
      sleep(5);
    }
  });
  dispatch_async(dispatch_get_global_queue(dispatch_queue_priority_default, 0), ^{
    sleep(1);
    @synchronized (_testlock) {
      
      [_testlock method2];
    }
  });
}

对于synchorize指令中使用的testlock为该锁标示,只有标示相同的时候才满足锁的效果。它的优点是不用显式地创建锁,便可以实现锁的机制。但是它会隐式地添加异常处理程序来保护代码,该程序在抛出异常的时候自动释放锁。

方式3 使用gcd

- (void)gcddemo {
  _testlock = [[testlock alloc] init];
  dispatch_semaphore_t semaphore = dispatch_semaphore_create(1);
  dispatch_async(dispatch_get_global_queue(dispatch_queue_priority_default, 0), ^{
    dispatch_semaphore_wait(semaphore, dispatch_time_forever);
    [_testlock method1];
    sleep(5);
    dispatch_semaphore_signal(semaphore);
  });
  dispatch_async(dispatch_get_global_queue(dispatch_queue_priority_default, 0), ^{
    sleep(1);
    dispatch_semaphore_wait(semaphore, dispatch_time_forever);
    [_testlock method2];
    dispatch_semaphore_signal(semaphore);
  });

}

方式4 使用phtread

- (void)pthreaddemo {
  _testlock = [[testlock alloc] init];
  
  __block pthread_mutex_t mutex;
  pthread_mutex_init(&mutex, null);
  
  //线程1
  dispatch_async(dispatch_get_global_queue(dispatch_queue_priority_default, 0), ^{
    pthread_mutex_lock(&mutex);
    [_testlock method1];
    sleep(5);
    pthread_mutex_unlock(&mutex);
  });
  
  //线程2
  dispatch_async(dispatch_get_global_queue(dispatch_queue_priority_default, 0), ^{
    sleep(1);
    pthread_mutex_lock(&mutex);
    [_testlock method2];
    pthread_mutex_unlock(&mutex);
  });
}

pthread_mutex_t定义在pthread.h,所以记得#include。

3 性能对比

这里简单写一个小程序来进行四种方式的性能对比,这里再固定次数内进行了加锁解锁,然后输出用时,结果如下(测试1、2执行次数不一样:测试1 < 测试2):

测试1

2016-11-05 15:27:52.595 lockdemo[4394:202297] nslock times:0.871843
2016-11-05 15:27:56.335 lockdemo[4394:202297] synthorize times:3.738939
2016-11-05 15:27:56.691 lockdemo[4394:202297] gcd times:0.355344
2016-11-05 15:27:57.328 lockdemo[4394:202297] pthread times:0.636815
2016-11-05 15:27:57.559 lockdemo[4394:202297] osspinlock times:0.231013
2016-11-05 15:27:57.910 lockdemo[4394:202297] os_unfair_lock times:0.350615

测试2

2016-11-05 15:30:54.123 lockdemo[4454:205180] nslock times:1.908103
2016-11-05 15:31:02.112 lockdemo[4454:205180] synthorize times:7.988547
2016-11-05 15:31:02.905 lockdemo[4454:205180] gcd times:0.792113
2016-11-05 15:31:04.372 lockdemo[4454:205180] pthread times:1.466987
2016-11-05 15:31:04.870 lockdemo[4454:205180] osspinlock times:0.497487
2016-11-05 15:31:05.637 lockdemo[4454:205180] os_unfair_lock times:0.767569

这里还测试了osspinlock(此类已经被os_unfair_lock所替代)。结果如下:

synthorize > nslock > pthread > gcd > os_unfair_lock >osspinlock

这里:

synthorize内部会添加异常处理,所以耗时。

pthread_mutex底层api,处理能力不错。

gcd系统封装的c代码效果比pthread好。

4 总结

简单就介绍这么多。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持移动技术网。

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

  • ios uicollectionview实现横向滚动

    现在使用卡片效果的app很多,之前公司让实现一种卡片效果,就写了一篇关于实现卡片的文章。文章最后附有demo实现上我选择了使用uicollectionview ... [阅读全文]
  • iOS UICollectionView实现横向滑动

    本文实例为大家分享了ios uicollectionview实现横向滑动的具体代码,供大家参考,具体内容如下uicollectionview的横向滚动,目前我使... [阅读全文]
  • iOS13适配深色模式(Dark Mode)的实现

    iOS13适配深色模式(Dark Mode)的实现

    好像大概也许是一年前, mac os系统发布了深色模式外观, 看着挺刺激, 时至今日用着也还挺爽的终于, 随着iphone11等新手机的发售, ios 13系统... [阅读全文]
  • ios 使用xcode11 新建项目工程的步骤详解

    ios 使用xcode11 新建项目工程的步骤详解

    xcode11新建项目工程,新增了scenedelegate这个类,转而将原appdelegate负责的对ui生命周期的处理担子接了过来。故此可以理解为:ios... [阅读全文]
  • iOS实现转盘效果

    本文实例为大家分享了ios实现转盘效果的具体代码,供大家参考,具体内容如下demo下载地址: ios转盘效果功能:实现了常用的ios转盘效果,轮盘抽奖效果的实现... [阅读全文]
  • iOS开发实现转盘功能

    本文实例为大家分享了ios实现转盘功能的具体代码,供大家参考,具体内容如下今天给同学们讲解一下一个转盘选号的功能,直接上代码直接看viewcontroller#... [阅读全文]
  • iOS实现轮盘动态效果

    本文实例为大家分享了ios实现轮盘动态效果的具体代码,供大家参考,具体内容如下一个常用的绘图,主要用来打分之类的动画,效果如下。主要是ios的绘图和动画,本来想... [阅读全文]
  • iOS实现九宫格连线手势解锁

    本文实例为大家分享了ios实现九宫格连线手势解锁的具体代码,供大家参考,具体内容如下demo下载地址:效果图:核心代码://// clockview.m// 手... [阅读全文]
  • iOS实现卡片堆叠效果

    本文实例为大家分享了ios实现卡片堆叠效果的具体代码,供大家参考,具体内容如下如图,这就是最终效果。去年安卓5.0发布的时候,当我看到安卓全新的material... [阅读全文]
  • iOS利用余弦函数实现卡片浏览工具

    iOS利用余弦函数实现卡片浏览工具

    本文实例为大家分享了ios利用余弦函数实现卡片浏览工具的具体代码,供大家参考,具体内容如下一、实现效果通过拖拽屏幕实现卡片移动,左右两侧的卡片随着拖动变小,中间... [阅读全文]
验证码:
移动技术网