Module

作者: 忧郁的河蟹 | 来源:发表于2015-05-16 09:47 被阅读133次

module是node.js最常用的模块,是node.js的根基,主要作用是导入模块,组织模块。

简单原理

module里面核心的方法是require,以下我将会用简单的伪代码来描述require的过程


var cache = {}; //模块缓存
function require(path) {

    var filename = _resolveFilename(path); // 读取模块的文件名字
    if (cache[filename]) {  //判断该模块是否已经加载到内存,如果已经在内存,则直接读取
        return cache[filename];
    }

    if (Native.module[filename]) { //判断模块是否原生 native的模块,如果是native模块,读取模块内容
        return Native.module[filename];
    }


    var content = fs.readFileSync(filename); // 用同步阻塞的方法读取文件内容

    var extension = getFileextension(filename); //根据文件extension的类型,根据不同的类型对不同文件进行相应的处理。

    if (extension == '.json') {
        return JSON.parse(content);
    }

    if (extension == '.node') {
        return process.dlopen; 
    }

    if (extension == '.js') {

        if (Module._contextLoad) { //根据_contextLoad 这个全局参数来区分,模块是需要所有文件都加载统一的上下文里面,还是分开不同上下文加载,通过exports的方法来跟其他模块通讯。node.js默认使用后者来处理。
            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: filename });

        } else {
            global.require = require;
            global.exports = self.exports;
            global.__filename = filename;
            global.__dirname = dirname;
            global.module = self;
            return runInThisContext(content, { filename: filename });

        }
    }
}

后记

总的来说,module模块还是比较简单,不过在浏览器里面就不是那么好实现了,根源还是在加载文件的时候,是否异步方法相关,尝试想想如果我们在加载模块的时候,使用fs.readFile 而不是readFileSync ,而执行一个文件当中,我们需要判断模块是否已经加载完毕,复杂度就变得大很多。

相关文章

网友评论

      本文标题:Module

      本文链接:https://www.haomeiwen.com/subject/zokkqttx.html