回顧 上篇文章大概展現了kmdjs0.1.x時期的編程範式: 以下面所示,能夠直接依賴注入到function裏,php
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加上來調用,如:node
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,自動匹配依賴
以下所示:git
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依賴分析和代碼重構github
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,能夠對代碼就行變換。如:express
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就行:app
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
轉載於猿2048:➮《kmdjs集成uglifyjs2打造極致的編程體驗》dom