Nodejs require 로드 메커니즘 (모듈 은 전체 공간 을 오염 시 킬 수 있 습 니 다)

23602 단어 JavaScriptNode.js
예전 에 Nodejs 의 MooTools 라 이브 러 리 가 이상 하 다 고 생각 했 습 니 다. 그 를 사용 할 때 require 의 반환 값 을 저장 할 필요 가 없 었 기 때문에 오늘 은 정말 참 을 수 없어 서 연 구 를 했 습 니 다. NodeJs 의 require 체제 에 대해 깊 은 이 해 를 가지 게 되 었 습 니 다.
MooTools 라 이브 러 리 의 "이상 한" 용법:
require('mootools');

  var QueryCommand = new Class({
        initialize: function (product, week, action) {
            this.product = product;
            this.weeknum = week;
            this.action = action;
        },

        toJsonString: function () {
            return JSON.stringify(this);
        }
    });

보 셨 습 니까? require 를 C + 의 include 로 사 용 했 습 니 다. 그 다음 코드 는 Class 형식 을 직접 사용 할 수 있 습 니 다.사실 MooTools 의 방법 은 Class 라 는 유형 을 전역 적 인 역할 영역 에 추가 하 는 것 이다.나 는 MooTools 가 이렇게 설계 되 었 을 것 이 라 고 생각한다. 왜냐하면 그 는 OO 라 이브 러 리 이기 때문이다.
그렇다면 무 툴 스 는 어떻게 자신 을 전역 적 인 역할 영역 에 부가 시 켰 을 까?사실은 매우 교묘 하 다. 편지 의 숫자 면 을 통 해 만 든 것 이다.
(function(){

var Class = this.Class = new Type('Class', function(params){
	if (instanceOf(params, Function)) params = {initialize: params};

	var newClass = function(){
		reset(this);
		if (newClass.$prototyping) return this;
		this.$caller = null;
		var value = (this.initialize) ? this.initialize.apply(this, arguments) : this;
		this.$caller = this.caller = null;
		return value;
	}.extend(this).implement(params);

	newClass.$constructor = Class;
	newClass.prototype.$constructor = newClass;
	newClass.prototype.parent = parent;

	return newClass;
});

......

})();

그 는 편지 의 숫자 면 량 을 통 해 익명 함 수 를 정의 한 후에 즉시 그것 을 집행 했다.이것 은 JS 의 관용 법 이다.한 가지 주의해 야 할 것 은 이 함 수 를 실행 할 때 사실은 this 지침 이 없다 는 것 이다!그러면 함수 내부 에서 this 를 사용 하면 전체 대상 에 접근 하 는 것 이 바로 그 원조 대상 입 니 다!
더 간단 한 예 를 보 자.
app.js
require('./a.js')
console.log(a);

a.js
a = 3; //  3

위 에 var a = 3 이 아 닙 니 다.그래서 이 문 구 는 전체 역할 영역 에 a 라 는 속성 을 추가 하 였 습 니 다 (주의, this 도 아 닙 니 다! 원조 대상 입 니 다). 그리고 app. js 모듈 에서 사용 할 수 있 습 니 다.사실 다른 모듈 에서 도 접근 할 수 있 습 니 다.
다음은 왜 전역 역할 영역 에 오염 되 었 는 지, nodejs 는 require 로 각 모듈 을 불 러 오지 않 았 습 니까? 모듈 은 서로 독립 되 어야 합 니 다. 사실 nodeJs 의 모듈 로 딩 순 서 는 이 렇 습 니 다.
Module._load("xxx.js") --> var module = new Module(); --> module.load("xxx.js") --> module._copile () -- > 최종 호출 된 것 은 wrapper 의 모듈 코드 입 니 다. 위의 예 를 들 면 최종 적 으로 실 행 된 것 은:
(function (exports, require, module, __filename, __dirname) { 
    a = 3;
});

module. copile () 의 코드 를 보면 알 수 있 습 니 다.
// Returns exception if any
Module.prototype._compile = function(content, filename) {
  var self = this;
  // remove shebang
  content = content.replace(/^\#\!.*/, '');

  function require(path) {
    return self.require(path);
  }

  require.resolve = function(request) {
    return Module._resolveFilename(request, self);
  };

  Object.defineProperty(require, 'paths', { get: function() {
    throw new Error('require.paths is removed. Use ' +
                    'node_modules folders, or the NODE_PATH ' +
                    'environment variable instead.');
  }});

  require.main = process.mainModule;

  // Enable support to add extra extension types
  require.extensions = Module._extensions;
  require.registerExtension = function() {
    throw new Error('require.registerExtension() removed. Use ' +
                    'require.extensions instead.');
  };

  require.cache = Module._cache;

  var dirname = path.dirname(filename);

  if (Module._contextLoad) {
    if (self.id !== '.') {
      debug('load submodule');
      // not root module
      var sandbox = {};
      for (var k in global) {
        sandbox[k] = global[k];
      }
      sandbox.require = require;
      sandbox.exports = self.exports;
      sandbox.__filename = filename;
      sandbox.__dirname = dirname;
      sandbox.module = self;
      sandbox.global = sandbox;
      sandbox.root = root;

      return runInNewContext(content, sandbox, filename, true);
    }

    debug('load root module');
    // root module
    global.require = require;
    global.exports = self.exports;
    global.__filename = filename;
    global.__dirname = dirname;
    global.module = self;

    return runInThisContext(content, filename, true);
  }

  // create wrapper function
  var wrapper = Module.wrap(content);

  var compiledWrapper = runInThisContext(wrapper, filename, true);
  if (global.v8debug) {
    if (!resolvedArgv) {
      // we enter the repl if we're not given a filename argument.
      if (process.argv[1]) {
        resolvedArgv = Module._resolveFilename(process.argv[1], null);
      } else {
        resolvedArgv = 'repl';
      }
    }

    // Set breakpoint on module start
    if (filename === resolvedArgv) {
      global.v8debug.Debug.setBreakPoint(compiledWrapper, 0, 0);
    }
  }
  var args = [self.exports, require, self, filename, dirname];
  return compiledWrapper.apply(self.exports, args);
};

Module._contextLoad 는 false 이기 때문에 일부 코드 는 무시 할 수 있 지만 몇 가지 중점 이 있 습 니 다.
  • 함수 입구 에서 require 함 수 를 정 의 했 고 require 함 수 는 정적 속성, 구성원 함수 도 있 습 니 다.최종 적 으로 require 는 copiled Wrapper. apply () 를 호출 하 는 실제 인삼 입 니 다.
  • var wrapper 는 문자열 형식의 변수 입 니 다. 값 은 "(function (exports, require, module, filename, dirname) {" + 당신 의 모듈 코드 + "}" 입 니 다. 바로 위 에 제 가 붙 인 포장 함수 입 니 다.결국, 그 는 runInThisContext () 를 호출 하여 문자열 을 진정한 함수 로 바 꾸 었 다.
  • 최종 적 으로 포 장 된 함 수 를 호출 하여 모듈 코드 를 실행 합 니 다.this 포인터 가 가리 키 는 것 은 self. exports, 즉 module. exports, 즉 exports 입 니 다.그래서 모듈 내부: this = = exports = = = module. exports.
  • args 는 매개 변수 입 니 다. 모듈 내부 에서 require () 를 사용 할 수 있 는 이 유 는 그 가 매개 변수 로 들 어 왔 기 때 문 입 니 다.

  • exports 는 사실 빈 대상 이기 때문에 exports 의 초기 값 은 비어 있 습 니 다. 이 코드 를 볼 수 있 습 니 다.
    function Module(id, parent) {
      this.id = id;
      this.exports = {};
      this.parent = parent;
      if (parent && parent.children) {
        parent.children.push(this);
      }
    
      this.filename = null;
      this.loaded = false;
      this.children = [];
    }

    마지막 으로 exports 를 어떻게 되 돌려 줍 니까? 이 함수 안에 있 습 니 다.
    Module._load = function(request, parent, isMain) {
      if (parent) {
        debug('Module._load REQUEST  ' + (request) + ' parent: ' + parent.id);
      }
    
      var filename = Module._resolveFilename(request, parent);
    
      var cachedModule = Module._cache[filename];
      if (cachedModule) {
        return cachedModule.exports;
      }
    
      if (NativeModule.exists(filename)) {
        // REPL is a special case, because it needs the real require.
        if (filename == 'repl') {
          var replModule = new Module('repl');
          replModule._compile(NativeModule.getSource('repl'), 'repl.js');
          NativeModule._cache.repl = replModule;
          return replModule.exports;
        }
    
        debug('load native module ' + request);
        return NativeModule.require(filename);
      }
    
      var module = new Module(filename, parent);
    
      if (isMain) {
        process.mainModule = module;
        module.id = '.';
      }
    
      Module._cache[filename] = module;
    
      var hadException = true;
    
      try {
        module.load(filename);
        hadException = false;
      } finally {
        if (hadException) {
          delete Module._cache[filename];
        }
      }
    
      return module.exports;
    };

    NodeJs 디 버 깅 이 쉽 지 않 습 니 다. 모든 코드 를 붙 여 놓 으 세 요.
    (function (exports, require, module, __filename, __dirname) { // Copyright Joyent, Inc. and other Node contributors.
    //
    // Permission is hereby granted, free of charge, to any person obtaining a
    // copy of this software and associated documentation files (the
    // "Software"), to deal in the Software without restriction, including
    // without limitation the rights to use, copy, modify, merge, publish,
    // distribute, sublicense, and/or sell copies of the Software, and to permit
    // persons to whom the Software is furnished to do so, subject to the
    // following conditions:
    //
    // The above copyright notice and this permission notice shall be included
    // in all copies or substantial portions of the Software.
    //
    // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
    // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
    // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
    // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
    // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
    // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
    // USE OR OTHER DEALINGS IN THE SOFTWARE.
    
    var NativeModule = require('native_module');
    var Script = process.binding('evals').NodeScript;
    var runInThisContext = Script.runInThisContext;
    var runInNewContext = Script.runInNewContext;
    var assert = require('assert').ok;
    
    
    // If obj.hasOwnProperty has been overridden, then calling
    // obj.hasOwnProperty(prop) will break.
    // See: https://github.com/joyent/node/issues/1707
    function hasOwnProperty(obj, prop) {
      return Object.prototype.hasOwnProperty.call(obj, prop);
    }
    
    
    function Module(id, parent) {
      this.id = id;
      this.exports = {};
      this.parent = parent;
      if (parent && parent.children) {
        parent.children.push(this);
      }
    
      this.filename = null;
      this.loaded = false;
      this.children = [];
    }
    module.exports = Module;
    
    // Set the environ variable NODE_MODULE_CONTEXTS=1 to make node load all
    // modules in thier own context.
    Module._contextLoad = (+process.env['NODE_MODULE_CONTEXTS'] > 0);
    Module._cache = {};
    Module._pathCache = {};
    Module._extensions = {};
    var modulePaths = [];
    Module.globalPaths = [];
    
    Module.wrapper = NativeModule.wrapper;
    Module.wrap = NativeModule.wrap;
    
    var path = NativeModule.require('path');
    
    Module._debug = function() {};
    if (process.env.NODE_DEBUG && /module/.test(process.env.NODE_DEBUG)) {
      Module._debug = function(x) {
        console.error(x);
      };
    }
    
    
    // We use this alias for the preprocessor that filters it out
    var debug = Module._debug;
    
    
    // given a module name, and a list of paths to test, returns the first
    // matching file in the following precedence.
    //
    // require("a.")
    //   -> a.
    //
    // require("a")
    //   -> a
    //   -> a.
    //   -> a/index.
    
    function statPath(path) {
      var fs = NativeModule.require('fs');
      try {
        return fs.statSync(path);
      } catch (ex) {}
      return false;
    }
    
    // check if the directory is a package.json dir
    var packageCache = {};
    
    function readPackage(requestPath) {
      if (hasOwnProperty(packageCache, requestPath)) {
        return packageCache[requestPath];
      }
    
      var fs = NativeModule.require('fs');
      try {
        var jsonPath = path.resolve(requestPath, 'package.json');
        var json = fs.readFileSync(jsonPath, 'utf8');
      } catch (e) {
        return false;
      }
    
      try {
        var pkg = packageCache[requestPath] = JSON.parse(json);
      } catch (e) {
        e.path = jsonPath;
        e.message = 'Error parsing ' + jsonPath + ': ' + e.message;
        throw e;
      }
      return pkg;
    }
    
    function tryPackage(requestPath, exts) {
      var pkg = readPackage(requestPath);
    
      if (!pkg || !pkg.main) return false;
    
      var filename = path.resolve(requestPath, pkg.main);
      return tryFile(filename) || tryExtensions(filename, exts) ||
             tryExtensions(path.resolve(filename, 'index'), exts);
    }
    
    // In order to minimize unnecessary lstat() calls,
    // this cache is a list of known-real paths.
    // Set to an empty object to reset.
    Module._realpathCache = {};
    
    // check if the file exists and is not a directory
    function tryFile(requestPath) {
      var fs = NativeModule.require('fs');
      var stats = statPath(requestPath);
      if (stats && !stats.isDirectory()) {
        return fs.realpathSync(requestPath, Module._realpathCache);
      }
      return false;
    }
    
    // given a path check a the file exists with any of the set extensions
    function tryExtensions(p, exts) {
      for (var i = 0, EL = exts.length; i < EL; i++) {
        var filename = tryFile(p + exts[i]);
    
        if (filename) {
          return filename;
        }
      }
      return false;
    }
    
    
    Module._findPath = function(request, paths) {
      var exts = Object.keys(Module._extensions);
    
      if (request.charAt(0) === '/') {
        paths = [''];
      }
    
      var trailingSlash = (request.slice(-1) === '/');
    
      var cacheKey = JSON.stringify({request: request, paths: paths});
      if (Module._pathCache[cacheKey]) {
        return Module._pathCache[cacheKey];
      }
    
      // For each path
      for (var i = 0, PL = paths.length; i < PL; i++) {
        var basePath = path.resolve(paths[i], request);
        var filename;
    
        if (!trailingSlash) {
          // try to join the request to the path
          filename = tryFile(basePath);
    
          if (!filename && !trailingSlash) {
            // try it with each of the extensions
            filename = tryExtensions(basePath, exts);
          }
        }
    
        if (!filename) {
          filename = tryPackage(basePath, exts);
        }
    
        if (!filename) {
          // try it with each of the extensions at "index"
          filename = tryExtensions(path.resolve(basePath, 'index'), exts);
        }
    
        if (filename) {
          Module._pathCache[cacheKey] = filename;
          return filename;
        }
      }
      return false;
    };
    
    // 'from' is the __dirname of the module.
    Module._nodeModulePaths = function(from) {
      // guarantee that 'from' is absolute.
      from = path.resolve(from);
    
      // note: this approach *only* works when the path is guaranteed
      // to be absolute.  Doing a fully-edge-case-correct path.split
      // that works on both Windows and Posix is non-trivial.
      var splitRe = process.platform === 'win32' ? /[\/\\]/ : /\//;
      // yes, '/' works on both, but let's be a little canonical.
      var joiner = process.platform === 'win32' ? '\\' : '/';
      var paths = [];
      var parts = from.split(splitRe);
    
      for (var tip = parts.length - 1; tip >= 0; tip--) {
        // don't search in .../node_modules/node_modules
        if (parts[tip] === 'node_modules') continue;
        var dir = parts.slice(0, tip + 1).concat('node_modules').join(joiner);
        paths.push(dir);
      }
    
      return paths;
    };
    
    
    Module._resolveLookupPaths = function(request, parent) {
      if (NativeModule.exists(request)) {
        return [request, []];
      }
    
      var start = request.substring(0, 2);
      if (start !== './' && start !== '..') {
        var paths = modulePaths;
        if (parent) {
          if (!parent.paths) parent.paths = [];
          paths = parent.paths.concat(paths);
        }
        return [request, paths];
      }
    
      // with --eval, parent.id is not set and parent.filename is null
      if (!parent || !parent.id || !parent.filename) {
        // make require('./path/to/foo') work - normally the path is taken
        // from realpath(__filename) but with eval there is no filename
        var mainPaths = ['.'].concat(modulePaths);
        mainPaths = Module._nodeModulePaths('.').concat(mainPaths);
        return [request, mainPaths];
      }
    
      // Is the parent an index module?
      // We can assume the parent has a valid extension,
      // as it already has been accepted as a module.
      var isIndex = /^index\.\w+?$/.test(path.basename(parent.filename));
      var parentIdPath = isIndex ? parent.id : path.dirname(parent.id);
      var id = path.resolve(parentIdPath, request);
    
      // make sure require('./path') and require('path') get distinct ids, even
      // when called from the toplevel js file
      if (parentIdPath === '.' && id.indexOf('/') === -1) {
        id = './' + id;
      }
    
      debug('RELATIVE: requested:' + request +
            ' set ID to: ' + id + ' from ' + parent.id);
    
      return [id, [path.dirname(parent.filename)]];
    };
    
    
    Module._load = function(request, parent, isMain) {
      if (parent) {
        debug('Module._load REQUEST  ' + (request) + ' parent: ' + parent.id);
      }
    
      var filename = Module._resolveFilename(request, parent);
    
      var cachedModule = Module._cache[filename];
      if (cachedModule) {
        return cachedModule.exports;
      }
    
      if (NativeModule.exists(filename)) {
        // REPL is a special case, because it needs the real require.
        if (filename == 'repl') {
          var replModule = new Module('repl');
          replModule._compile(NativeModule.getSource('repl'), 'repl.js');
          NativeModule._cache.repl = replModule;
          return replModule.exports;
        }
    
        debug('load native module ' + request);
        return NativeModule.require(filename);
      }
    
      var module = new Module(filename, parent);
    
      if (isMain) {
        process.mainModule = module;
        module.id = '.';
      }
    
      Module._cache[filename] = module;
    
      var hadException = true;
    
      try {
        module.load(filename);
        hadException = false;
      } finally {
        if (hadException) {
          delete Module._cache[filename];
        }
      }
    
      return module.exports;
    };
    
    Module._resolveFilename = function(request, parent) {
      if (NativeModule.exists(request)) {
        return request;
      }
    
      var resolvedModule = Module._resolveLookupPaths(request, parent);
      var id = resolvedModule[0];
      var paths = resolvedModule[1];
    
      // look up the filename first, since that's the cache key.
      debug('looking for ' + JSON.stringify(id) +
            ' in ' + JSON.stringify(paths));
    
      var filename = Module._findPath(request, paths);
      if (!filename) {
        var err = new Error("Cannot find module '" + request + "'");
        err.code = 'MODULE_NOT_FOUND';
        throw err;
      }
      return filename;
    };
    
    
    Module.prototype.load = function(filename) {
      debug('load ' + JSON.stringify(filename) +
            ' for module ' + JSON.stringify(this.id));
    
      assert(!this.loaded);
      this.filename = filename;
      this.paths = Module._nodeModulePaths(path.dirname(filename));
    
      var extension = path.extname(filename) || '.js';
      if (!Module._extensions[extension]) extension = '.js';
      Module._extensions[extension](this, filename);
      this.loaded = true;
    };
    
    
    Module.prototype.require = function(path) {
      assert(typeof path === 'string', 'path must be a string');
      assert(path, 'missing path');
      return Module._load(path, this);
    };
    
    
    // Resolved path to process.argv[1] will be lazily placed here
    // (needed for setting breakpoint when called with --debug-brk)
    var resolvedArgv;
    
    
    // Returns exception if any
    Module.prototype._compile = function(content, filename) {
      var self = this;
      // remove shebang
      content = content.replace(/^\#\!.*/, '');
    
      function require(path) {
        return self.require(path);
      }
    
      require.resolve = function(request) {
        return Module._resolveFilename(request, self);
      };
    
      Object.defineProperty(require, 'paths', { get: function() {
        throw new Error('require.paths is removed. Use ' +
                        'node_modules folders, or the NODE_PATH ' +
                        'environment variable instead.');
      }});
    
      require.main = process.mainModule;
    
      // Enable support to add extra extension types
      require.extensions = Module._extensions;
      require.registerExtension = function() {
        throw new Error('require.registerExtension() removed. Use ' +
                        'require.extensions instead.');
      };
    
      require.cache = Module._cache;
    
      var dirname = path.dirname(filename);
    
      if (Module._contextLoad) {
        if (self.id !== '.') {
          debug('load submodule');
          // not root module
          var sandbox = {};
          for (var k in global) {
            sandbox[k] = global[k];
          }
          sandbox.require = require;
          sandbox.exports = self.exports;
          sandbox.__filename = filename;
          sandbox.__dirname = dirname;
          sandbox.module = self;
          sandbox.global = sandbox;
          sandbox.root = root;
    
          return runInNewContext(content, sandbox, filename, true);
        }
    
        debug('load root module');
        // root module
        global.require = require;
        global.exports = self.exports;
        global.__filename = filename;
        global.__dirname = dirname;
        global.module = self;
    
        return runInThisContext(content, filename, true);
      }
    
      // create wrapper function
      var wrapper = Module.wrap(content);
    
      var compiledWrapper = runInThisContext(wrapper, filename, true);
      if (global.v8debug) {
        if (!resolvedArgv) {
          // we enter the repl if we're not given a filename argument.
          if (process.argv[1]) {
            resolvedArgv = Module._resolveFilename(process.argv[1], null);
          } else {
            resolvedArgv = 'repl';
          }
        }
    
        // Set breakpoint on module start
        if (filename === resolvedArgv) {
          global.v8debug.Debug.setBreakPoint(compiledWrapper, 0, 0);
        }
      }
      var args = [self.exports, require, self, filename, dirname];
      return compiledWrapper.apply(self.exports, args);
    };
    
    
    function stripBOM(content) {
      // Remove byte order marker. This catches EF BB BF (the UTF-8 BOM)
      // because the buffer-to-string conversion in `fs.readFileSync()`
      // translates it to FEFF, the UTF-16 BOM.
      if (content.charCodeAt(0) === 0xFEFF) {
        content = content.slice(1);
      }
      return content;
    }
    
    
    // Native extension for .js
    Module._extensions['.js'] = function(module, filename) {
      var content = NativeModule.require('fs').readFileSync(filename, 'utf8');
      module._compile(stripBOM(content), filename);
    };
    
    
    // Native extension for .json
    Module._extensions['.json'] = function(module, filename) {
      var content = NativeModule.require('fs').readFileSync(filename, 'utf8');
      try {
        module.exports = JSON.parse(stripBOM(content));
      } catch (err) {
        err.message = filename + ': ' + err.message;
        throw err;
      }
    };
    
    
    //Native extension for .node
    Module._extensions['.node'] = process.dlopen;
    
    
    // bootstrap main module.
    Module.runMain = function() {
      // Load the main module--the command line argument.
      Module._load(process.argv[1], null, true);
      // Handle any nextTicks added in the first tick of the program
      process._tickCallback();
    };
    
    Module._initPaths = function() {
      var isWindows = process.platform === 'win32';
    
      if (isWindows) {
        var homeDir = process.env.USERPROFILE;
      } else {
        var homeDir = process.env.HOME;
      }
    
      var paths = [path.resolve(process.execPath, '..', '..', 'lib', 'node')];
    
      if (homeDir) {
        paths.unshift(path.resolve(homeDir, '.node_libraries'));
        paths.unshift(path.resolve(homeDir, '.node_modules'));
      }
    
      if (process.env['NODE_PATH']) {
        var splitter = isWindows ? ';' : ':';
        paths = process.env['NODE_PATH'].split(splitter).concat(paths);
      }
    
      modulePaths = paths;
    
      // clone as a read-only copy, for introspection.
      Module.globalPaths = modulePaths.slice(0);
    };
    
    // bootstrap repl
    Module.requireRepl = function() {
      return Module._load('repl', '.');
    };
    
    Module._initPaths();
    
    // backwards compatibility
    Module.Module = Module;
    
    });

    References:
  • runInThisContext: http://www.cnblogs.com/rubylouvre/archive/2011/11/25/2262521.html
  • Script. runInThis Context 및 nodejs requiree: http://andy0807.iteye.com/blog/1446769
  • 잘못된 것 이나 시대 에 뒤떨어 진 MooTools 오염 또는 전체 역할 영역 오염 방지 에 관 한 글 을 보면 MooTools 가 전체 역할 영역 을 오염 시 키 는 것 이 고의 일 수 있 음 을 알 수 있다.http://davidwalsh.name/mootools-nodejs
  • 좋은 웹페이지 즐겨찾기