flowJS源码个人分析
刚刚在腾讯云技术社区前端专栏中看到一篇腾讯高级前端工程师写的《一个只有99行代码的js流程框架》觉得很屌,感觉是将后台的简单的工作流思维搬到了前端js实现,本人不才在这里拜读解析下源码,而且经常有新手问我的很多问题其实是不懂如何调试一段js代码,在这这里就详细说明下我是怎么调试flowJS的源码思路的。前端大神见笑小弟的总结
首先第一段:
var obj = arguments[0], flow = {init: function(){}}, flowData = {}, noop = function(){}, init = 'init', trace = flowJS.trace = flowJS.trace||[init];
这段代码就是把要用到的变量在开始声明用逗号隔开,为了代码的整洁和节省代码的语句数量。
第二段:
if(({}).toString.call(obj) === '[object Object]'){//判断第一个参数是否是对象,这个方法最准确
extend(flow, obj);
flow.init.call(extend({getCurr:function(){
return init;
}, stepData:function(dataName){
return dataName?undefined:{};
}, getPrev:noop, fail:noop, success:noop}, new Step(init)));
}
读到extend(flow,obj);我们应该猜到是将参数和flow合并,我们在来看extend方法:
function prop(obj, fun){
for(var p in obj) {
obj.hasOwnProperty(p) && fun(p);//判断这个属性是原始属性还是原型上的属性
}
}
/*对象的合并扩展*/
function extend(des, src){
prop(src, function(p){
des[p] = src[p];
});
return des;
}
extend的方法是将src的原生属性添加到des对象中。然后我们接着主线看:
flow.init.call(extend({getCurr:function(){
return init;
}, stepData:function(dataName){
return dataName?undefined:{};
}, getPrev:noop, fail:noop, success:noop}, new Step(init)));
这段代码就是要执行参数中的init方法并且将一个自定义添加Step(init)返回的对象中的属性的对象作为init中的this对象。那重点在Step方法,但由于里面有点复杂刚开始接触你看不到他在里什么的各种方法的用处,在这里我们最后结合《一个只有99行代码的js流程框架》文章提供的api例子来调试里面各个方法的作用这样更好的梳理整个功能的运行逻辑,在这里就体现了张镇圳大神的所说的一个模块的功能中代码的逻辑梳理、可读性、语义化多么重要,但flowJS这个框架可以理解为项目的在js原生上封装了一层脱了业务层的框架,这种框架一般是组长或是高级前端工程来维护,而且一般随着时间的积累慢慢成熟成型后期的改动会很小,所以这里只要相对用面向对象梳理下逻辑就可以,而业务层随客户的不断增多,需求的改动频繁出现,而且项目的大部分改动工作量都在业务层上,所以业务层代码逻辑梳理、可读性、语义化尤为重要,关系到后期维护的成本。
我们先从文章中第一个例子来调试;
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>flowJS demo1</title>
</head>
<body>
</body>
<script type="text/javascript" src="./flowJS.js"></script>
<script type="text/javascript">
flowJS({
init:function(){
this.setNext('步骤A').setNext('步骤B').setNext('步骤C'); //预先设置好需要经过的每个步骤
this.next();
},
'步骤A':function(){
console.log('执行 步骤A');
this.next(); //告诉流程执行下一步
},
'步骤B':function(){
console.log('执行 步骤B');
this.next();
},
'步骤C':function(){
console.log('执行 步骤C');
}
});
</script>
</html>
第一例子中就两个函数一个setNext,next;我们先来看setNext方法:
this.setNext = function(stepName, s, f){
success = s||success;//获取参数中成功回调函数
fail = f||fail;//获取参数中失败回调函数
nextStepName = stepName||nextStepName;//判断stepName是否为空,为空将赋值原来的
nextStep = new Step(nextStepName);//实例化对象
return nextStep;
};
比较简单,就是实例化step对象。之后我们在看next方法代码:
this.next = function(stepName, s, f){
nextStepName = stepName||nextStepName;
success = s||success;
fail = f||fail;
nextStep = stepName?new Step(stepName):nextStep;
if(nextStepName){
nextStep.stepData = function(dataName){
return dataName?nextData[dataName]:nextData;
};
nextStep.getPrev = function(){
return name;
};
nextStep.fail = function(){
fail.apply(this, arguments);
};
nextStep.success = function(){
success.apply(this, arguments);
};
stepMapping[this.getCurr()] = true;
if(allDone()){
trace.push(nextStepName);
typeof nextStepName == 'string' && proxy(nextStepName);
if(({}).toString.call(nextStepName) === '[object Array]'){
for(var i=0; i<nextStepName.length; i++){
proxy(nextStepName[i]);
}
}
function proxy(stepName, fn){
if(typeof (fn = flow[stepName]) == 'function'){
(function(fn, context){
return function(){
debugger;
return fn.apply(context, arguments);
};
})(fn, extend({getCurr:function(){
return stepName;
}}, nextStep))();
}else{
throw new Error("step not found: "+stepName);
}
}
}
return nextStep;
}
};
一开始懵逼看不懂很正常,我们在里面加断点跟着demo一步一步走对整个逻辑的理解会更清晰一点。
this.next = function(stepName, s, f){
debugger;
nextStepName = stepName||nextStepName;
success = s||success;
fail = f||fail;
nextStep = stepName?new Step(stepName):nextStep;
if(nextStepName){
nextStep.stepData = function(dataName){
在里面加了debugger,然后我们一步一步的调试:

开始我觉得遗憾的是为什么nextStepName里的值是“步骤A”,整个方法是在这个地方调用的。

而nextStepName的变量是在setNext复制的

按道理应该里面的值是最后一次调用setNext的参数也就是“步骤C”,那为什么运行结果里面的值是步骤A呢,这里要看你的理解领悟能了,算法有了,结果有了,你能强行理解算法而经历往结果靠拢么?这个问题的重点是在setNext的返回值,setNext返回的是新的step实例对象。在结合
this.setNext('步骤A').setNext('步骤B').setNext('步骤C'); //预先设置好需要经过的每个步骤
用图来解析这个方法的输出和执行会更好理解:

this.next();
这段代码执行的this对象中的nextStepName私有变量的值为步骤A。
然后我们接着往下看next方法里的代码:
success = s||success;//赋值参数的成功回调函数
fail = f||fail;//赋值参数的失败回调函数
nextStep = stepName?new Step(stepName):nextStep;//如果第一参数有值的话返回这个值得step实例对象,没有的话返回对象的nextStep私有变量
if(nextStepName){
/*对象上添加一系列方法 暂时忽略*/
nextStep.stepData = function(dataName){
return dataName?nextData[dataName]:nextData;
};
nextStep.getPrev = function(){
return name;
};
nextStep.fail = function(){
fail.apply(this, arguments);
};
nextStep.success = function(){
success.apply(this, arguments);
};
/*对象上添加一系列方法 暂时忽略*/
stepMapping[this.getCurr()] = true;//标记上一步步骤被执行了
代码加了注释基本很好理解就不多说了。下一个重点是这句
if(allDone()){
下回分解。
源码可以在腾讯云技术社区上下载,我把源码都上加注释在放到Github上,下载暂时不放上去。
flowJS源码个人分析的更多相关文章
- MapReduce的ReduceTask任务的运行源码级分析
MapReduce的MapTask任务的运行源码级分析 这篇文章好不容易恢复了...谢天谢地...这篇文章讲了MapTask的执行流程.咱们这一节讲解ReduceTask的执行流程.ReduceTas ...
- Activity源码简要分析总结
Activity源码简要分析总结 摘自参考书籍,只列一下结论: 1. Activity的顶层View是DecorView,而我们在onCreate()方法中通过setContentView()设置的V ...
- MapReduce的MapTask任务的运行源码级分析
TaskTracker任务初始化及启动task源码级分析 这篇文章中分析了任务的启动,每个task都会使用一个进程占用一个JVM来执行,org.apache.hadoop.mapred.Child方法 ...
- TaskTracker任务初始化及启动task源码级分析
在监听器初始化Job.JobTracker相应TaskTracker心跳.调度器分配task源码级分析中我们分析的Tasktracker发送心跳的机制,这一节我们分析TaskTracker接受JobT ...
- MongoDB源码分析——mongod程序源码入口分析
Edit 说明:第一次写笔记,之前都是看别人写的,觉得很简单,开始写了之后才发现真的很难,不知道该怎么分析,这篇文章也参考了很多前辈对MongoDB源码的分析,也有一些自己的理解,后续将会继续分析其他 ...
- FFmpeg的HEVC解码器源码简单分析:解析器(Parser)部分
===================================================== HEVC源码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpeg ...
- FFmpeg源码简单分析:libswscale的sws_scale()
===================================================== FFmpeg的库函数源码分析文章列表: [架构图] FFmpeg源码结构图 - 解码 FFm ...
- LinkedHashMap 源码详细分析(JDK1.8)
1. 概述 LinkedHashMap 继承自 HashMap,在 HashMap 基础上,通过维护一条双向链表,解决了 HashMap 不能随时保持遍历顺序和插入顺序一致的问题.除此之外,Linke ...
- [Java] LinkedHashMap 源码简要分析
特点 * 各个元素不仅仅按照HashMap的结构存储,而且每个元素包含了before/after指针,通过一个头元素header,形成一个双向循环链表.使用循环链表,保存了元素插入的顺序. * 可设置 ...
随机推荐
- 每天一个Linux命令(09)--touch命令
linux的touch命令不常用,一般在使用make的时候可能会用到,用来修改文件时间戳,或者新建一个不存在的文件. 1.命令格式: touch [选项]··· 文件··· 2.命令参数: -a 或 ...
- 最简单的 RabbitMQ 监控方法 - 每天5分钟玩转 OpenStack(158)
这是 OpenStack 实施经验分享系列的第 8 篇. 先来看张图:这是 Nova 的架构图,我们可以看到有两个组件处于架构的中心位置:数据库和Queue.数据库保存状态信息,而几乎所有的 nova ...
- Nginx rewrite(重读)
Nginx Rewrite规则相关指令 Nginx Rewrite规则相关指令有if.rewrite.set.return.break等,其中rewrite是最关键的指令.一个简单的Nginx Re ...
- Sublime Text 3 (Build 3126) 最新注册码
Sublime Text 作为程序员开发神器,听说最新版更新了 并且增加了不少新特性.马上到官网下载了最新版 Sublime Text 3 3126 使用了下,反应速度比以前的确更快了.随手找了几个S ...
- 2017-3-5 C#基础 函数
函数/方法:非常抽象独立完成某项功能的一个个体 函数的作用: 提高代码的重用性提高功能开发的效率提高程序代码的可维护性 函数分为: 固定功能函数高度抽象函数 函数四要素:输入,输出,函数体,函数名 p ...
- 做推送,怎么能不了解推送的 4 种消息形式呢?(iOS 篇)
极光推送是为 App 提供第三方推送服务的平台之一,它提供四种消息形式:通知,自定义消息,富媒体和本地通知.笔者将基于官方说明与个人理解来谈一下这四种消息.本篇为 iOS 篇,Android 篇入口. ...
- Spring 3.0 Aop 入门
关于Aop的原理,动态代理,反射,之类的底层java技术网上搜一堆一堆的..我就不多说了,主要说在spring上使用aop的方法. 首先不得不说一下的就是,spring aop的支持需要外部依赖包: ...
- Troubleshooting OpenStack Bug- 每天5分钟玩转 OpenStack(162)
这是 OpenStack 实施经验分享系列的第 12 篇. 问题描述 客户报告了一个问题:对 instance 执行 migrate 操作,几个小时了一直无法完成,不太正常. 问题分析 遇到这种情况, ...
- 在.NET项目中使用PostSharp,使用CacheManager实现多种缓存框架的处理
在前面几篇随笔中,介绍了PostSharp的使用,以及整合MemoryCache,<在.NET项目中使用PostSharp,实现AOP面向切面编程处理>.<在.NET项目中使用Pos ...
- 一个蛋疼的CTF图片隐写
话不多说,直接上原题 TIPS:心中无码 打开解题链接,是一张png图片,直接用16进制编辑器打开,没有附加其它文件.看下文件区段信息也很正常. 又拖进stegsolve,Blue的0位很不正常 多次 ...