node.js 动态执行脚本
node.js最近新增了虚拟机模块,其实也不能说是新增的,只是把一些内部接口暴露出来罢了,从2.x就有了。我们可以从node/src/node.js看到这些代码:
varScript=process.binding('evals').NodeScript; varrunInThisContext=Script.runInThisContext; NativeModule.wrap=function(script){ returnNativeModule.wrapper[0]+script+NativeModule.wrapper[1]; }; NativeModule.wrapper=[ '(function(exports,require,module,__filename,__dirname){', '\n});' ]; NativeModule.prototype.compile=function(){ varsource=NativeModule.getSource(this.id); source=NativeModule.wrap(source); varfn=runInThisContext(source,this.filename,true); fn(this.exports,NativeModule.require,this,this.filename); this.loaded=true; };
其中的Script对象,就与require('vm')返回的对象很相似,而实质上,vm模块就是对Script对象的封装。
varScript=process.binding('evals').NodeScript; console.log(Script) /** {[Function:NodeScript] createContext:[Function], runInContext:[Function], runInThisContext:[Function], runInNewContext:[Function]} */ console.log(require('vm')) {Script: {[Function:NodeScript] createContext:[Function], runInContext:[Function], runInThisContext:[Function], runInNewContext:[Function]}, createScript:[Function], createContext:[Function], runInContext:[Function], runInThisContext:[Function], runInNewContext:[Function]}
其中,runInThisContext相当于一个全新的环境中执行代码,不会影响当前作用域的对象。而runInNewContext与runInContext则能指定是上下文对象,区别是一个普通对象或一个context对象。换言之,runInNewContext与runInContext能局部影响当前作用域的对象。要与当前环境完全进行交互的话,就需要用到危险的eval。在node.js自带的加载体系中,显然没有这样的勇气,使用的是runInThisContext。并且在这之前做了许多工作,如把用户的JS文件里面的内容再包一层(NativeModule.wrap),还有其他凌散操作,加之是同步操作,实际上是一种效率很糟的加载方式。唯一的好处是,使用了同步,让代码编写起来简单多了。
在github中,已有人对这几种动态执行脚本的方法进行性能比较:
varvm=require('vm'), code='varsquare=n*n;', fn=newFunction('n',code), script=vm.createScript(code), sandbox; n=5; sandbox={n:n}; benchmark=function(title,funk){ varend,i,start; start=newDate; for(i=0;i<5000;i++){ funk(); } end=newDate; console.log(title+':'+(end-start)+'ms'); } varctx=vm.createContext(sandbox); benchmark('vm.runInThisContext',function(){vm.runInThisContext(code);}); benchmark('vm.runInNewContext',function(){vm.runInNewContext(code,sandbox);}); benchmark('script.runInThisContext',function(){script.runInThisContext();}); benchmark('script.runInNewContext',function(){script.runInNewContext(sandbox);}); benchmark('script.runInContext',function(){script.runInContext(ctx);}); benchmark('fn',function(){fn(n);}); /** vm.runInThisContext:212ms vm.runInNewContext:2222ms script.runInThisContext:6ms script.runInNewContext:1876ms script.runInContext:44ms fn:0ms */
由此可见,还是v8自带的方法Function完胜!
以上就是本文的全部内容,希望能给大家一个参考,也希望大家多多支持毛票票。