title: Node.js 源码分析 - 加载 js 文件
date: 2018-11-30 21:04:49
tags:
- Node.js
- Node.js 源码分析
- 源码分析
categories:
- Node.js 源码分析
此文最初于四年前发布在个人站上的,现迁移至此重发,原链接:https://laogen.site/nodejs/nodejs-src/bootstrap-js/
《Node.js 源码分析》 系列目录页:https://laogen.site/nodejs/nodejs-src/index/
了解 js 文件加载前的准备工作
在《从 main 函数开始》这篇中说到了 LoadEnvironment()
函数负责加载 js 代码,但并没有继续说明加载细节。
这篇从 LoadEnvironment()
开始探究 js 代码加载的详细过程。
LoadEnvironment()
的逻辑分两部分:
loaders.js
node.js
,执行后得到两个启动函数;这段代码比较长,我们把不影响主逻辑的代码省略掉,然后直接在代码中以注释的形式来解释:
void LoadEnvironment(Environment* env) { // ... /************************************************************/ /**** 第一步.加载并执行两个 js 文件:`loaders.js` `node.js`****/ /************************************************************/ // The bootstrapper scripts are lib/internal/bootstrap/loaders.js and // lib/internal/bootstrap/node.js, each included as a static C string // defined in node_javascript.h, generated in node_javascript.cc by // node_js2c. // 这两个 js 文件在 node 构建过程中就被转换成了 C++ 代码,即以 C++ 字符串的 // 形式存在于 C++ 代码中,根据这个文件名就可以直接获取相应的 js 代码字符串; // loaders.js 的文件名 Local<String> loaders_name = FIXED_ONE_BYTE_STRING(env->isolate(), "internal/bootstrap/loaders.js"); // 执行 loaders.js 得到函数: `loaders_bootstrapper` MaybeLocal<Function> loaders_bootstrapper = GetBootstrapper(env, LoadersBootstrapperSource(env), loaders_name); // node.js 文件名 Local<String> node_name = FIXED_ONE_BYTE_STRING(env->isolate(), "internal/bootstrap/node.js"); // 执行 loaders.js 得到函数: `loaders_bootstrapper` MaybeLocal<Function> node_bootstrapper = GetBootstrapper(env, NodeBootstrapperSource(env), node_name); // 上面代码中:LoadersBootstrapperSource() & NodeBootstrapperSource() 是 // 在 /src/node_javascript.h 头文件中声明的,node 源码中并没有它们的具体实现, // 它们的实现代码是在 node 本身构建过程中生成的; // 至于 GetBootstrapper(),它的作用是编译&执行 js 代码,返回执行结果。 if (loaders_bootstrapper.IsEmpty() || node_bootstrapper.IsEmpty()) { return; } Local<Object> global = env->context()->Global(); // ... // Expose the global object as a property on itself // (Allows you to set stuff on `global` from anywhere in JavaScript.) global->Set(FIXED_ONE_BYTE_STRING(env->isolate(), "global"), global); /*************************************************************************/ /* 第二步.分别调用这两个启动函数:loaders_bootstrapper、node_bootstrapper ****/ /*************************************************************************/ // Create binding loaders // 基于 GetBinding() 函数模板 创建 get_binding_fn 函数 Local<Function> get_binding_fn = env->NewFunctionTemplate(GetBinding)->GetFunction(env->context()) .ToLocalChecked(); // 基于 GetLinkedBinding() 函数模板 创建 get_linked_binding_fn 函数 Local<Function> get_linked_binding_fn = env->NewFunctionTemplate(GetLinkedBinding)->GetFunction(env->context()) .ToLocalChecked(); // 基于 GetInternalBinding() 函数模板 创建 get_internal_binding_fn 函数 Local<Function> get_internal_binding_fn = env->NewFunctionTemplate(GetInternalBinding)->GetFunction(env->context()) .ToLocalChecked(); // 上面三个函数会作为 调用 loaders_bootstrapper() 时的参数。 Local<Value> loaders_bootstrapper_args[] = { env->process_object(), get_binding_fn, get_linked_binding_fn, get_internal_binding_fn, Boolean::New(env->isolate(), env->options()->debug_options->break_node_first_line) }; // loaders_bootstrapper() 调用结果将保存在这个变量, // 接下来,它将被作为参数传给另一个启动函数:node_bootstrapper() Local<Value> bootstrapped_loaders; // 调用启动函数 loaders_bootstrapper() if (!ExecuteBootstrapper(env, loaders_bootstrapper.ToLocalChecked(), arraysize(loaders_bootstrapper_args), loaders_bootstrapper_args, &bootstrapped_loaders)) { return; } // Bootstrap Node.js Local<Object> bootstrapper = Object::New(env->isolate()); SetupBootstrapObject(env, bootstrapper); Local<Value> bootstrapped_node; Local<Value> node_bootstrapper_args[] = { env->process_object(), bootstrapper, bootstrapped_loaders }; // 调用启动函数 loaders_bootstrapper() if (!ExecuteBootstrapper(env, node_bootstrapper.ToLocalChecked(), arraysize(node_bootstrapper_args), node_bootstrapper_args, &bootstrapped_node)) { return; } }
LoadEnvironment()
主要是调用了两个 启动函数(Bootstrapper)
其中 loaders_bootstrapper() 主要实现了一个简单的模块加载机制名为 NativeModule
,主要用于加载内部模块的,会在 node_bootstrapper() 中用到;
而在 node_bootstrapper() 则加载并执行了用户的 js 文件(也就是通常的 app.js 或 index.js)。
这两个启动函数分别定义在 /lib/internal/bootstrap/loaders.js
和 /lib/internal/bootstrap/node.js
文件中;
接下来的两篇文,会分别对这两个文件进行详细的探究,弄清楚 js 文件加载执行的细节;
Maslow (wangfugen@126.com), laf.js 作者。