在Appium1.7.1里集成了一个同步模块async-lock用来支持多会话功能。

只能说就算是以单线程高并发闻名的I/O密集型Nodejs也不得不扩展额外的同步块方法,或者说,在现有的计算机体系结构和配备的操作系统之下,所有的编程语言都无法摒弃同步信息块。

不过想想也是,这个世界原本是处于无序的状态,只是有了人类的干预,才会让一切事情直着走。要是没有同步块,所有的线程(Nodejs底层的异步也是多线程)就会如野草生成一般,力量强大,但不可被控制,而不可被控制恰恰是不能被人类所容忍的。

首先需要明白一点的是,或者说以下内容的前提是,Nodejs碰到异步回调会启动I/O,之后继续往下执行。

以官网的两句代码(https://www.npmjs.com/package/async-lock)为例:

user1和user1操作redis数据库,取同一个键的值乘以2:

User1
lock.acquire('key', function(cb){
// Concurrency safe
redis.get('key', function(err, value){
redis.set('key', value * 2, cb);
});
}, function(err, ret){
});
User2
lock.acquire('key', function(cb){
// Concurrency safe
redis.get('key', function(err, value){
redis.set('key', value * 2, cb);
});
}, function(err, ret){
});

acquire函数的签名如下:

AsyncLock.prototype.acquire = function (key, fn, cb, opts)

key:就是”key”

fn:就是执行的函数

cb:fn函数执行完后的回调函数

opts:配置参数

再看acquire函数的具体实现

对于User1,进入acquire函数执行的是:

这里是个关键:queue是一个字典型,一开始key为空,queue[‘key’]已经被赋值空数组了,执行exec(true)函数。

exec函数:

if else判断需要返回的是Promise,还是以函数回调的方式结束

User1、User2的例子是函数回调方式,所以进入if条件执行。

cb是fn(cb)传过来的,在执行完fn()函数后,就会调用cb:

这里是重点,在调用fn之前,都是CPU在执行,现在执行fn函数,会有回调,这个时候异步I/O启动,CPU继续往下执行,这里的CPU往下执行不是进入fn函数体,而是执行User2的代码,即开始执行:

还是进入acquire函数,这个时候走的是另一个分支:

把exec函数存进一个数组,然后User2就执行完了。CPU继续往下执行User2之后的代码。

假如在某个时刻,User1的异步执行到了cb函数,即红框内的函数:

会调用done函数,其中的关键如下:

可以看到,User2被移出,并且执行其中的exec函数,这样就能保证User2的异步代码在User1之后执行。

总结起来,是以一个中间共有的队列,存放异步函数,在上一个操作完临界资源后,再从队列里shift出来执行异步函数,以达到同步的目的。

async-lock模块理解的更多相关文章

  1. C# async 和 await 理解

    C# async 和 await 理解 先假设如下场景: 主函数 Main,循环等待用户输入: 计算函数 Cal,耗时计算大量数据: class Test { static int Main(stri ...

  2. generator 到 async 的简单理解。

    generator 到 async 的简单理解.觉得实现方式很有意思. 1. generator generator 函数返回一个遍历器对象 遍历器对象 每次调用next 方法 返回 有着value ...

  3. async/await 深度理解使用

    在vue中使用 eg async created () { await setTimeout(()=>{ console.log(1) },5000); }, async mounted () ...

  4. 万向节锁(Gimbal Lock)的理解

    [TOC] 结论 我直接抛出结论: Gimbal Lock 产生的原因不是欧拉角也不是旋转顺序,而是我們的思维方式和程序的执行逻辑没有对应,也就是说是我们的观念导致这个情况的发生. 他人解释 首先我们 ...

  5. await和async更多的理解

    最近有不少网友提起await和async,呵呵,C# 5引进的语法糖. 这个语法糖还真不好吃,能绕倒一堆初学的朋友,在网上也有很多网友关于这块知识点的争论,有对有错,今天在这里把这个误区好好讲讲. 在 ...

  6. async和await理解代码

    <1>:Async和Await的理解1 using System; using System.Collections.Generic; using System.Linq; using S ...

  7. volatile和synchronized与lock的理解

    volatile 特征: a:可见性:一个线程修改了某个共享变量的值,其他线程能够立马得知这个修改. b:禁止特定的处理器重排序. volatile的内存语义: 1.当写一个volatile变量的时候 ...

  8. HTTP模块理解(二)

    这是我在写,用express+ajax+swig来做一个简单的应用的时候,遇到的问题.还是不太理解http模块. 后来在网上看到云栖社区的一篇<Node.js之HTTP请求与响应>,这里做 ...

  9. Node.js的http模块理解

    Node.js标准库提供了http模块,其中封装了一个高效的HTTP服务器和一个简易的HTTP客户端. http.Server是一个基于事件的HTTP服务器,它的核心由C++编写,兼顾高性能和简易性 ...

随机推荐

  1. Akka系列---什么是Actor

    本文已.Net语法为主,同时写有Scala及Java实现代码 严肃的说,演员是一个广泛的概念,作为外行人我对Actor 模型的定义: Actor是一个系统中参与者的虚拟人物,Actor与Actor之间 ...

  2. Scrum 项目4.0&&5.0

    MY—HR 成员: 角色分配 学号 博客园 4.0团队贡献分 5.0团队贡献分 丘惠敏 PM项目经理 201406114203 http://www.cnblogs.com/qiuhuimin/ 19 ...

  3. 【第二周】PSP

    日期 C类别 C内容 S开始时间 E结束时间 I间隔(单位:分钟) T净时间(单位:分钟) 9月8日 编程 结对编程 12:15 13:15 10 50    编程 结对编程  16:35 17:30 ...

  4. c++的继承方式

    c++的继承,因为学完过的时间太长,忘了,现在再温习一下. c++的继承方式 1. 公有继承(public) 2. 私有继承(private) 3. 保护继承(protected) 从一个基类派生的继 ...

  5. HDU 2061 Treasure the new start, freshmen!

    http://acm.hdu.edu.cn/showproblem.php?pid=2061 Problem Description background:A new semester comes , ...

  6. maven执行"mvn clean package" 命令报错

    昨天利用mvn打包,执行程序'mvn clean package' 命令,发现打包失败 问题描述 具体看代码 发信tomcat下的log 清除不掉.为什么呢?忽然想起来我的项目服务还起着,于是我把服务 ...

  7. 【设计模式】—— 原型模式Prototype

    前言:[模式总览]——————————by xingoo 模式意图 由于有些时候,需要在运行时指定对象时哪个类的实例,此时用工厂模式就有些力不从心了.通过原型模式就可以通过拷贝函数clone一个原有的 ...

  8. ZOJ2760_How Many Shortest Path

    给一个图,求从某个点到另一个点的最短路有多少条?所有的路都不共边. 首先从终点开始Spfa标记最短距离,然后建图. 建图的时候,如果满足两点之间的最短路只差为两点之间的边长,那么在网络流的模型中连接一 ...

  9. 【uoj#315/bzoj4943】[NOI2017]蚯蚓排队 Hash

    题目描述 给出 $n$ 个字符,初始每个字符单独成字符串.支持 $m$ 次操作,每次为一下三种之一: $1\ i\ j$ :将以 $i$ 结尾的串和以 $j$ 开头的串连到一起. $2\ i$ :将 ...

  10. 更新ffmpeg

    今天对公司线上的几台机器做了下ffmpeg的更新,没有什么技术含量,还是简单记录下,做个流水账~哈哈 软件包获取方式 官方网站:https://ffmpeg.org/download.htmlgith ...