co模块整体包括三部分

  1. 对于几种参数类型的判断,主要判断是否object,array,promise,generator,generatorFunction这几种;

  2. 将几种不同的参数类型转换为promise

  3. 递归执行promise

     module.exports = co['default'] = co.co = co;
    let slice = [].slice; // 多次用到,所以定义成变量; 原文是Array.prototype.slice // 1. 对于几种参数类型的判断,主要判断是否object,array,promise,generator,generatorFunction这几种; /**
    * 判断是否为对象
    * 任何数据类型的constructor属性返回创建该对象的函数的引用,实例的constructor属性指向其构造函数,
    */
    function isObject(obj) {
    return obj.constructor === Object;
    } /**
    * 是否promise
    * promise具有.then方法,.catch方法
    */
    function isPromise(obj) {
    return 'function' === typeof obj.then; //也可以用 obj.constructor === Promise
    } /**
    * 是否generator对象,
    * generator具有next,throw,return方法
    */
    function isGenerator(obj) {
    return 'function' === typeof obj.next() && 'function' === typeof obj.throw() && 'function' === typeof obj.return();
    } /**
    * generator的constructor的name为GeneratorFunction,displayName(非标准)如果设置也为* GeneratorFunction,
    * display相关参考https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Function/displayName
    *【注意】generator的constructor !== GeneratorFunction , 参考自这里https://blog.csdn.net/sinat_30552885/article/details/85247120
    */
    function isGeneratorFunction(obj) {
    let constructor = obj.constructor;
    if ( !constructor ) return false;
    if ( constructor.name === 'GeneratorFunction' || constructor.displayName === 'GeneratorFunction') return true;
    return isGenerator(obj);
    } // 2. 将几种不同的参数类型转换为promise // 总括
    function toPromise(obj) {
    if (!obj) return obj;
    if ( isPromise(obj) ) return obj;
    if ( isGenerator(obj) || isGeneratorFunction(obj) ) return co.call(this, obj); // co的入参接收promise或者generator,因此如果是generator,则直接调用一遍co;
    if ( 'function' === typeof obj) return thunkToPromise.call(this, obj);
    if ( Array.isArray(obj) ) return arrayToPromise.call(this, obj);
    if ( isObject(obj) ) return objToPromise.call(this, obj);
    return obj;
    } /**
    * thunk => promise
    */
    function thunkToPromise(fn) {
    let ctx = this;
    return new Promise(function(resolve, reject) {
    fn.call(ctx, function(err, res) {
    if ( err ) reject(err);
    if ( arguments.length > 2) res = slice.call(arguments,1);
    resolve(res);
    })
    })
    } /**
    * arr => promise
    * 将数组的每一元素变成promise,然后放入到promise.all里统一存储
    */
    function arrayToPromise(arr) {
    return Promise.all(arr.map(toPromise, this));
    } /**
    * obj => promise;
    * obj.constructor 返回创建该对象的函数的引用
    */
    function objToPromise(obj) {
    let results = new obj.constructor();
    let promises = [];
    let keys = Object.keys(obj);
    for ( let i = 0; i < keys.length; i++ ) {
    let key = keys[i];
    let promise = toPromise(this, obj[key]);
    if ( promise && isPromise(promise) ) {
    defer(promise, key);
    } else {
    results[key] = obj[key];
    }
    }
    return Promise.all(promises).then(function() {
    return results;
    })
    function defer(promise, key) {
    promises.push(promise.then(function(res) {
    results[key] = res;
    }))
    }
    } // 3. 递归执行promise /**
    *
    * @param {*} fn 入参是一个promise或者generator
    * @return Promise;
    */
    function co(gen) {
    let ctx = this;
    let args = slice.call(arguments, 1);
    return new Promise(function(resolve, reject) {
    if ( typeof gen === 'function') gen = gen.apply(ctx, args);
    if ( !gen || typeof gen !== 'function' ) resolve(gen);
    onFulfilled();
    function onFulfilled() {
    let ret;
    try {
    ret = gen.next();
    } catch (e) {
    return reject(e);
    }
    next(ret);
    }
    function onRejected() {
    let ret;
    try {
    ret = gen.throw();
    } catch (e) {
    return reject(e)
    }
    next(ret);
    }
    function next(ret) {
    if ( ret.done ) return resolve(ret.value);
    let value = toPromise.call(this, ret.value);
    if ( value && isPromise(value)) return value.then(onFulfilled, onRejected);
    } return onRejected(new TypeError('You may only yield a function, promise, generator, array, or object, but the following object was passed: "' + String(ret.value) + '"'));
    })
    } co.wrapper = function(fn) {
    createPromise.__generatorFunction__ = fn;
    return createPromise;
    function createPromise() {
    return co.call(this, fn.apply(this,arguments))
    }
    }

co源码的更多相关文章

  1. 【原】Android热更新开源项目Tinker源码解析系列之三:so热更新

    本系列将从以下三个方面对Tinker进行源码解析: Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Android热更新开源项目Tinker源码解析系列之二:资源文件热更新 A ...

  2. C# ini文件操作【源码下载】

    介绍C#如何对ini文件进行读写操作,C#可以通过调用[kernel32.dll]文件中的 WritePrivateProfileString()和GetPrivateProfileString()函 ...

  3. 【原】FMDB源码阅读(三)

    [原]FMDB源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 FMDB比较优秀的地方就在于对多线程的处理.所以这一篇主要是研究FMDB的多线程处理的实现.而 ...

  4. 从源码看Azkaban作业流下发过程

    上一篇零散地罗列了看源码时记录的一些类的信息,这篇完整介绍一个作业流在Azkaban中的执行过程,希望可以帮助刚刚接手Azkaban相关工作的开发.测试. 一.Azkaban简介 Azkaban作为开 ...

  5. 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新

    [原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...

  6. 【原】Android热更新开源项目Tinker源码解析系列之二:资源文件热更新

    上一篇文章介绍了Dex文件的热更新流程,本文将会分析Tinker中对资源文件的热更新流程. 同Dex,资源文件的热更新同样包括三个部分:资源补丁生成,资源补丁合成及资源补丁加载. 本系列将从以下三个方 ...

  7. 多线程爬坑之路-Thread和Runable源码解析之基本方法的运用实例

    前面的文章:多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类) 多线程爬坑之路-Thread和Runable源码解析 前面 ...

  8. SDWebImage源码解读之SDWebImageDownloaderOperation

    第七篇 前言 本篇文章主要讲解下载操作的相关知识,SDWebImageDownloaderOperation的主要任务是把一张图片从服务器下载到内存中.下载数据并不难,如何对下载这一系列的任务进行设计 ...

  9. 【深入浅出jQuery】源码浅析--整体架构

    最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...

  10. ABP源码分析一:整体项目结构及目录

    ABP是一套非常优秀的web应用程序架构,适合用来搭建集中式架构的web应用程序. 整个Abp的Infrastructure是以Abp这个package为核心模块(core)+15个模块(module ...

随机推荐

  1. 牛客练习赛3 绝对半径 ——k尺取法

    题目 链接 题意:一个n个数字的序列,最多去掉其中k个,使得连续相同数字序列的长度尽量长 分析 如果不考虑去掉元素,就是普通的尺取.考虑到去元素,则只需做一点修改. 先离散化,再把每种元素的位置用va ...

  2. PHP mysqli_get_host_info() 函数

    定义和用法 mysqli_get_host_info() 函数返回 MySQL 服务器主机名和连接类型. 语法 mysqli_get_host_info(connection); 返回 MySQL 服 ...

  3. diff:二进制文件内容差异比较

    在Ubuntu 18.04下验证,造冰箱的大熊猫@cnblogs 2019/7/29 假设我们需要以二进制格式比较两个文件file1.bin和file2.bin的差异,一个简单的方法是 1)先使用xx ...

  4. OpenSSL 通过OCSP手动验证证书

    翻译:https://raymii.org/s/articles/OpenSSL_Manually_Verify_a_certificate_against_an_OCSP.html?utm_sour ...

  5. HDU 5863 cjj's string game ( 16年多校10 G 题、矩阵快速幂优化线性递推DP )

    题目链接 题意 : 有种不同的字符,每种字符有无限个,要求用这k种字符构造两个长度为n的字符串a和b,使得a串和b串的最长公共部分长度恰为m,问方案数 分析 : 直觉是DP 不过当时看到 n 很大.但 ...

  6. 灰度变换,gama变换,对数,反对数变换

    学习DIP第2天 灰度变换,及按照一定规则对像素点的灰度值进行变换,变换的结果可以增强对比度,或者达到其他的效果(例如二值化,或者伽马变换),由于灰度变换为针对单个像素点的灰度值进行变换,素以算法复杂 ...

  7. Java学习笔记(持续更新ing)

    1.在读入字符串时:    str = sc.nextLine();     //读入一行                                     str = sc.next();   ...

  8. [Linux]安装kali虚拟机后忘记root密码

    1. 登录时,按e进入编辑模式 2. 编辑模式 修改 ro 修改为 rw 添加 init=/bin/bash 修改完按 F10 3. 选择recovery mode 回车 4.输入命令passwd 设 ...

  9. skbuff

    在2.6.24之后这个结构体有了较大的变化,此处先说一说2.6.16版本的sk_buff,以及解释一些问题. 一. 先直观的看一下这个结构体~~~~~~~~~~~~~~~~~~~~~~在下面解释每个字 ...

  10. 2018-2019-2 20165330《网络对抗技术》Exp9 Web安全基础

    目录 基础问题 实验目的 实验内容 实验步骤 实验总结与体会 实验目的 本实践的目标理解常用网络攻击技术的基本原理. 返回目录 实验内容 WebGoat准备工作 SQL注入攻击 命令注入(Comman ...