kmdjs集成uglifyjs2打造极致的编程体验
回顾
上篇文章大概展示了kmdjs0.1.x时期的编程范式:
如下面所示,可以直接依赖注入到function里,
kmdjs.define('main',['util.bom','app.Ball','util.dom.test'], function(bom,Ball,test) {
var ball = new Ball(0, 0, 28, 1, -2, 'kmdjs');
var vp = bom.getViewport();
});
也可以直接在代码里把full namespace加上来调用,如:
kmdjs.define('main',['util.bom','app.Ball','util.dom.test'], function() {
var ball = new app.Ball(0, 0, 28, 1, -2, 'kmdjs');
var vp = util.bom.getViewport();
});
而且,在循环依赖的场景,因为执行顺序的问题,会导致第一种方式注入undefined,所以循环依赖的情况下只能用full namespace的方式来调用。
这种编程体验虽然已经足够好,但是可以更好。怎样才算更好?
- 不用依赖注入function
- 不用写full namespace,自动匹配依赖
如下所示:
kmdjs.define('main',['util.bom','app.Ball','util.dom.test'], function() {
var ball = new Ball(0, 0, 28, 1, -2, 'kmdjs');
var vp = bom.getViewport();
});
这就要借助uglifyjs能力,把function的字符串替换成带有namespace就可以实现上面的效果。
uglifyjs依赖分析和代码重构
function fixDeps(fn,deps) {
var U2 = UglifyJS;
//uglify2不支持匿名转ast
var code = fn.toString().replace('function','function ___kmdjs_temp');
var ast = U2.parse(code);
ast.figure_out_scope();
var nodes = [];
ast.walk(new U2.TreeWalker(function (node) {
if (node instanceof U2.AST_New) {
var ex = node.expression;
var name = ex.name;
isInWindow(name) || isInArray(nodes, node) || isInScopeChainVariables(ex.scope, name) || nodes.push({name:name,node:node});
}
if (node instanceof U2.AST_Dot) {
var ex = node.expression;
var name = ex.name;
var scope = ex.scope;
if (scope) {
isInWindow(name) || isInArray(nodes, node) || isInScopeChainVariables(ex.scope, name) || nodes.push({name:name,node:node});
}
}
if (node instanceof U2.AST_SymbolRef) {
var name = node.name;
isInWindow(name) || isInArray(nodes, node) || isInScopeChainVariables(node.scope, name) || nodes.push({name:name,node:node});
}
}));
var cloneNodes = [].concat(nodes);
//过滤new nodes 中的symbo nodes
for (var i = 0, len = nodes.length; i < len; i++) {
var nodeA = nodes[i].node;
for (var j = 0, cLen = cloneNodes.length; j < cLen; j++) {
var nodeB = cloneNodes[j].node;
if (nodeB.expression === nodeA) {
nodes.splice(i, 1);
i--;
len--;
}
}
}
for (var i = nodes.length; --i >= 0;) {
var item = nodes[i],
node=item.node,
name=item.name;
var fullName=getFullName(deps,name);
var replacement;
if (node instanceof U2.AST_New) {
replacement = new U2.AST_New({
expression: new U2.AST_SymbolRef({
name:fullName
}),
args: node.args
});
} else if (node instanceof U2.AST_Dot) {
replacement = new U2.AST_Dot({
expression: new U2.AST_SymbolRef({
name: fullName
}),
property: node.property
});
}else if(node instanceof U2.AST_SymbolRef){
replacement = new U2.AST_SymbolRef({
name: fullName
});
}
var start_pos = node.start.pos;
var end_pos = node.end.endpos;
code = splice_string(code, start_pos, end_pos, replacement.print_to_string({
beautify: true
}));
}
return code.replace('function ___kmdjs_temp','function');
}
function getFullName(deps,name){
var i= 0,
len=deps.length,
matchCount= 0,
result=[];
for(;i<len;i++) {
var fullName = deps[i];
if (fullName.split('.').pop() === name) {
matchCount++;
if (!isInArray(result, fullName)) result.push(fullName);
}
}
if(matchCount>1){
throw "the same name conflict: "+result.join(" and ");
} else if(matchCount===1){
return result[0];
}else{
throw ' can not find module ['+name+']';
}
}
function splice_string(str, begin, end, replacement) {
return str.substr(0, begin) + replacement + str.substr(end);
}
function isInScopeChainVariables(scope, name) {
var vars = scope.variables._values;
if (Object.prototype.hasOwnProperty.call(vars, "$" + name)) {
return true;
}
if (scope.parent_scope) {
return isInScopeChainVariables(scope.parent_scope, name);
}
return false;
}
function isInArray(arr,name){
var i= 0,len=arr.length;
for(;i<len;i++){
if(arr[i]===name){
return true;
}
}
return false;
}
function isInWindow(name){
if(name==='this')return true;
return name in window;
}
通过上面的fixDeps,可以对代码就行变换。如:
console.log(fixDeps(function (A) {
var eee = m;
var b = new A();
var b = new B();
var c = new C();
var d = G.a;
},['c.B','AAA.G','SFSF.C','AAAA.m'] ))
输出:
function (A) {
var eee = AAAA.m;
var b = new A();
var b = new c.B();
var c = new SFSF.C();
var d = AAA.G.a;
}
这样,kmdjs在执行模块function的时候,只需要fixDeps加上full namespace就行:
function buildBundler(){
var topNsStr = "";
each(kmdjs.factories, function (item) {
nsToCode(item[0]);
});
topNsStr+= kmdjs.nsList.join('\n') +"\n\n";
each(kmdjs.factories, function (item) {
topNsStr+=item[0]+' = ('+ fixDeps(item[2],item[1])+')();\n\n' ;
});
if(kmdjs.buildEnd) kmdjs.buildEnd(topNsStr);
return topNsStr;
}
build出来的包,当然全都加上了namespace。再也不用区分循环依赖和非循环依赖了~~~
Github
上面的所有代码可以Github上找到:
https://github.com/kmdjs/kmdjs
kmdjs集成uglifyjs2打造极致的编程体验的更多相关文章
- EDM数据营销之电商篇| 六大事务性邮件,环环相扣打造极致用户体验!
“以用户为中心”的时代,电商们致力于打造极致的用户体验,想尽各式新颖营销办法,但难免还是会出现营销断层,以至于和用户间无法达到完整的交互. 本次Focussend以邮件营销为例,聚焦用户从浏览到支付等 ...
- Dora.Interception,为.NET Core度身打造的AOP框架 [1]:更加简练的编程体验
很久之前开发了一个名为Dora.Interception的开源AOP框架(github地址:https://github.com/jiangjinnan/Dora,如果你觉得这个这框架还有那么一点价值 ...
- observejs改善组件编程体验
传送门 observejs:https://github.com/kmdjs/observejs 本文演示:http://kmdjs.github.io/observejs/list/ 本文代码:ht ...
- 依赖注入[6]: .NET Core DI框架[编程体验]
毫不夸张地说,整个ASP.NET Core框架是建立在一个依赖注入框架之上的,它在应用启动时构建请求处理管道过程中,以及利用该管道处理每个请求过程中使用到的服务对象均来源于DI容器.该DI容器不仅为A ...
- iOS-Socket编程体验
CHENYILONG Blog Socket编程体验 Socket编程体验 技术博客http://www.cnblogs.com/ChenYilong/新浪微博http://weibo.com/lu ...
- AOP框架Dora.Interception 3.0 [1]: 编程体验
.NET Core正式发布之后,我为.NET Core度身定制的AOP框架Dora.Interception也升级到3.0.这个版本除了升级底层类库(.NET Standard 2.1)之外,我还对它 ...
- 全新升级的AOP框架Dora.Interception[1]: 编程体验
多年之前利用IL Emit写了一个名为Dora.Interception(github地址,觉得不错不妨给一颗星)的AOP框架.前几天利用Roslyn的Source Generator对自己为公司写的 ...
- .NET CORE学习笔记系列(2)——依赖注入[6]: .NET Core DI框架[编程体验]
原文https://www.cnblogs.com/artech/p/net-core-di-06.html 毫不夸张地说,整个ASP.NET Core框架是建立在一个依赖注入框架之上的,它在应用启动 ...
- AMD and CMD are dead之KMDjs集成Blob一键下载全部build包
更新 不zuo,[A/C]MD就不会死,所以kmdjs赢来来其伟大的版本0.0.6,该版本主要的更新有: 移除去了kmdjs.get(..).then的支持,只支持kmdjs.get(-,functi ...
随机推荐
- git如何切换远程仓库
场景 工作时可能由于git仓库的变动,需要我们将已有代码切换仓库.比如我们先用的gitlab,现在要切换到github上. 迁移命令 代码迁移其实也很简单. 先保证本地代码是最新代码 $ git pu ...
- nginx代理https站点(亲测)
nginx代理https站点(亲测) 首先,我相信大家已经搞定了nginx正常代理http站点的方法,下面重点介绍代理https站点的配置方法,以及注意事项,因为目前大部分站点有转换https的需要所 ...
- 一步步学习javascript基础篇(8):细说事件
终于学到事件了,不知道为何听到“事件”就有一种莫名的兴奋.可能是之前的那些知识点过于枯燥无味吧,说起事件感觉顿时高大上了.今天我们就来好好分析下这个高大上的东西. 可以说,如果没有事件我们的页面就只能 ...
- 玩转Windows服务系列——使用Boost.Application快速构建Windows服务
玩转Windows服务系列——创建Windows服务一文中,介绍了如何快速使用VS构建一个Windows服务.Debug.Release版本的注册和卸载,及其原理和服务运行.停止流程浅析分别介绍了Wi ...
- 自己开发实现OAuth做webapi认证
看到园子里面有人写的OAuth,就想把自己实现的OAuth也分享一下,关于OAuth协议这里就不再赘述. 一.作为认证服务器,首先需要提供一个可以通过appid/appsecret来获取token这样 ...
- 一文搞懂HMM(隐马尔可夫模型)
什么是熵(Entropy) 简单来说,熵是表示物质系统状态的一种度量,用它老表征系统的无序程度.熵越大,系统越无序,意味着系统结构和运动的不确定和无规则:反之,,熵越小,系统越有序,意味着具有确定和有 ...
- The type javax.ws.rs.core.MediaType cannot be resolved. It is indirectly referenced from required .class files
看到了http://stackoverflow.com/questions/5547162/eclipse-error-indirectly-referenced-from-required-clas ...
- Protobuf使用规范分享
一.Protobuf 的优点 Protobuf 有如 XML,不过它更小.更快.也更简单.它以高效的二进制方式存储,比 XML 小 3 到 10 倍,快 20 到 100 倍.你可以定义自己的数据结构 ...
- fir.im Weekly - APP 性能监测优化 二三事
每一个成功的 App,都拥有强大的性能体验.本期 fir.im Weekly 整理了微信读书.美团外卖. 天猫.美团点评技术团队的关于性能监测优化方面策略和工具的分享,一起来看看. 微信读书 iOS ...
- Android 开发环境在 Windows7 下的部署安装
Android SDK Android SDK 为 Android 应用的开发.测试和调试提了必要的API库和开发工具. ADT Bundle 下载 如果你是一个android 开发新手,推荐你下载使 ...