本篇博客我们来聊一下ReactiveSwift中的原子性操作,在此内容上我们简单的聊一下Posix互斥锁以及递归锁的概念以及使用场景。然后再聊一下Atomic的代码实现。Atomic主要负责多线程下的原子操作,负责共享资源的同步一致性。而在Atomic中就是使用到了Posix互斥锁和递归锁。在聊上述内容之前,我们先来回顾一下Swift语言中延迟执行defer的使用方式,在之前Swift编程的相关博客中也涉及到了defer的使用方式。defer因为Atomic使用到了延迟操作,所以下方我们再做一个defer的简单回顾。

一、Defer延迟执行

在本篇博客之所以聊defer延迟执行,是因为在Atomic的代码实现中,使用了defer{}来为操作加的锁。具体的说是在操作前进行加锁,然后紧接着使用defer{}进行解锁,稍后我们会进行介绍。Swift中的Defer延迟执行是比较常用的,其用法也是比较简单的。下方就是我们列举了一个示例,该示例比较简单。首先我们输出a,然后使用defer块执行输出b的语句,然后就是输出c,最后是使用defer块输出d,具体代码如下。

  

在看上述代码输出结果时,我们可以先预测一下输出结果,是输出“abcd”还是“acbc”还是“acdb”呢?下方就是上述代码片段的输出结果。从下方代码片段中我们不难看出,其结果为“acdb”。从结果中我们不难看出defer{}块的执行顺序是在该作用域结束之前从后往前执行。

  

根据上述代码片段以及输出结果,我们可以画出下方这个运行简图,如下所示。因为下方这个简图并不复杂,在此就不做过多赘述了。

  

二、Posix互斥锁

互斥锁的概念就不做过多赘述了,简单的说就是防止多个线程同时修改一块共享区域,导致数据不同步的情况发生而添加的锁。互斥锁的互斥是线程的互斥,也就是说添加互斥锁的代码块在同一个时间点上只允许一个线程对这块共享区域进行操作,其他线程若想对该块区域进行操作的话,需要等待之前的互斥锁打开后方可进入。

下方这个PosixThreadMutex类是Atomic的代码实现中封装的Posix互斥锁。在该代码实现中主要调用了C语言中的线程互斥锁的相关函数。下方锁的使用方式,稍后再聊Atomic类的时候会使用到下方的这个互斥锁。

  

三、递归锁

接下来我们来看一下递归锁,在Atomic的代码实现中也使用到了递归锁。“递归锁”顾名思义就是在递归中使用的锁,普通锁在一个线程中是不能被重用的,也就是说一个普通锁被上锁后,你就不能再次调用上锁的方法了,这样会出问题的。等普通锁被解锁后,你才可以对其进行上锁。针对普通锁的不可重用性,我们给出了下方示例,如下所示。

  

上述代码片段比较简单,对普通锁连续执行了两次lock,然后执行我们的代码块,之后就是执行了两次unlock。该代码使用普通锁的步骤其实是与递归函数中使用普通锁的场景是一样的。当递归实现函数是,执行第一次递归时,添加了一个普通锁,在锁未打开时,第二次递归时又会执行一下上锁。这种场景与上述代码片段是一样的。因为普通锁在同一个线程中的不可重用性,所以上述代码会产生死锁DeadLock。下方截图就是上述代码片段所执行的结果:

  

我们将上述代码中的NSLock普通锁修改成NSRecursiveLock递归锁后,如下。下方的代码就可以正确执行。因为递归锁可以在同一个线程中重复的使用。具体如下所示:

  

既然是递归锁,那么接下来我们就在递归函数中来使用一下递归锁。在递归函数执行使用锁时,本质上是多次使用一个锁。也就是在一个锁未上锁时再次对其上锁。下方就是递归锁在递归函数中使用的简单示例。

  

四、Atomic原子操作的具体代码实现

聊完互斥锁和递归锁后,接下来我们来看一下ReactiveSwift中的原子操作Atomic类的具体代码实现。Aotmic中使用了上述我们聊的Posix互斥锁来进行的原子操作。下方就是Atomic类的代码实现。

从下方代码片段中我们不难看出,在value读取和修改时都会进行加锁和解锁操作。当然,解锁的代码是放在defer{}的语句块中进行操作的。在Atomic.swift文件中还有一个RecursiveAtomic类,也就是负责递归原子操作的。RecursiveAtomic与Atomic类的不同之处在于前者使用的是递归锁,负责在递归操作时保持原子操作。而后者使用的是互斥锁。RecursiveAtomic的代码与Atomic类的代码类似,在此就不做过多赘述了。

  

今天的博客就先到这儿,下篇博客我们会继续解析ReactiveSwift框架中的其他内容。

上述代码github分享地址:https://github.com/lizelu/TipSwiftForRac

p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 13.0px Menlo; color: #ffffff }
span.s1 { }
p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 13.0px Menlo; color: #ffffff }
span.s1 { }

ReactiveSwift源码解析(十一) Atomic的代码实现以及其中的Defer延迟、Posix互斥锁、递归锁的更多相关文章

  1. ReactiveSwift源码解析(八) SignalProducer的代码的基本实现

    在前面几篇博客中我们详细的聊了ReactiveSwift中的Bag.Event.Observer以及Signal的使用方式和代码实现.那么在接下来的这几篇博客中,我们就依附于之前博客的基础上来聊一聊S ...

  2. ReactiveSwift源码解析(三) Signal代码的基本实现

    上篇博客我们详细的聊了ReactiveSwift源码中的Bag容器,详情请参见<ReactiveSwift源码解析之Bag容器>.本篇博客我们就来聊一下信号量,也就是Signal的的几种状 ...

  3. ReactiveSwift源码解析(五) SignalProtocol的observe()、Map、Filter延展实现

    上篇博客我们对Signal的基本实现以及Signal的面向协议扩展进行了介绍, 详细内容请移步于<Signal中的静态属性静态方法以及面向协议扩展>.并且聊了Signal的所有的g功能扩展 ...

  4. Laravel源码解析之model(代码)

    本篇文章给大家带来的内容是关于Laravel源码解析之model(代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助. 前言 提前预祝猿人们国庆快乐,吃好.喝好.玩好,我会在电视上看 ...

  5. ReactiveSwift源码解析(十二) MutableProperty基本代码实现

    前两篇博客我们分别聊了ReactiveSwift框架中的负责标记对象的生命周期的类Lifetime以及负责原子性操作的Atomic类的具体代码实现.前两篇博客之所以聊Lifetime以及Atomic的 ...

  6. ReactiveSwift源码解析(一) Event与Observer代码实现

    ReactiveCocoa这个框架是做什么用的本篇博客就不做过多赘述了,什么是"响应式编程"也不多聊了,自行Google吧.本篇博客的主题是解析ReactiveCocoa框架中的核 ...

  7. ReactiveSwift源码解析(二) Bag容器的代码实现

    今天博客我接着上篇博客的内容来,上篇博客我们详细的看了ReactiveSwift中的Observer已经Event的代码实现.接下来我们来看一下ReactiveSwift中的结构体Bag的实现.Bag ...

  8. ReactiveSwift源码解析(四) Signal中的静态属性静态方法以及面向协议扩展

    上篇博客我们聊了Signal的几种状态.Signal与Observer的关联方式以及Signal是如何向关联的Observer发送事件的.本篇博客继续上篇博客的内容,来聊一下Signal类中静态的ne ...

  9. ReactiveSwift源码解析(七) Signal的CombineLatest的代码实现

    本篇博客我们就来聊一下combineLatest()的使用以及具体的实现方式.在之前的<iOS开发之ReactiveCocoa下的MVVM>的博客中我们已经聊过combineLatest( ...

随机推荐

  1. Java多线程中join方法详解

    join()方法用于让当前执行线程等待join线程执行结束.其实现原理是不停的检查join线程是否存活,如果join线程存活则让当前线程永远等待. join()方法部分实现细节 while(isAli ...

  2. winform制作小工具的技巧

    在使用winfrom制作一些工具的时候,一些基本设置都是去属性里面找来找去,一段时间就忘了,记录记录以备不时之需. 一.窗体绘制的常用设置 窗体的设置应当在窗体构造函数中InitializeCompo ...

  3. MyBatis源码解析【4】反射和动态代理

    通过之前的介绍,我们了解了几个组件的生命周期. 它也是我们重要装备之一. 今天我们需要搞一件更加强的装备,叫做反射和动态代理. 如果没有这件装备的话,显然后面的源码boss是打不动的. 顺便说一下,下 ...

  4. Java 泛型在实际开发中的应用

    java泛型是对Java语言的类型系统的一种扩展,泛型的本质就是将所操作的数据类型参数化.下面我会由浅入深地介绍Java的泛型. 一:泛型出现的背景 在java代码里,你会经常发现类似下边的代码: p ...

  5. 跳跳棋(9018_1563)(BZOJ_2144)

    题目: Hzwer的跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子. 某一天,黄金大神和cjy用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在a,b,c这三个位置.他们要 ...

  6. 获取子物体数量---Transform.childCount

    如何判断一个物体下是否有子物体?getchild(0)!=null?显然不可取 那去获取拿到子物体数量?transform.GetChildCount();可以解决 但在新版本中已被弃用,可用tran ...

  7. 了解Java基础原理

    Java 是1995年SUN公司推出的一门高级编程语言,是面向互联网的语言,WEB应用程序首选的语言(安卓底层,大数据hadoop框架用java编写,Spark用Scala编写,Scala用java写 ...

  8. FreeRTOS——内存管理

    1. 标准malloc() 和 free() 库函数的缺陷: 1)在小型的嵌入式系统中,可能不可用. 2)具体实现相对较大,占用较多宝贵的代码空间. 3)通常不具备线程安全性. 4)具有不确定性,每次 ...

  9. linux用户及权限管理

    [文件管理.管道.用户及组管理.用户及权限管理]\用户及组管理 用户与组管理 Linux系统是一个多用户多任务的分时操作系统,任何一个要使用系统资源的用户,都必须首先向系统管理员申请一个账号,然后以这 ...

  10. Excel Countif函数用法

    公式:=COUNTIF(范围,条件) 例1: 统计A1:A11当中,等于数字3的单元格,结果是4个. 例2:还可以进行大于(>),大于等于(>=),小于(<),小于等于(<=) ...