成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Node.js源碼研究之模塊組織加載

開發 前端
粗略研究了一下node.js源碼,它有8000行C++代碼,2000行javascript代碼,來看看js和C++間是怎么組織連接起來,以及各個模塊是怎樣互相調用的。

粗略研究了一下node.js源碼,它有8000行C++代碼,2000行javascript代碼,來看看js和C++間是怎么組織連接起來,以及各個模塊是怎樣互相調用的。

本文使用的node.js版本是0.4.8,可以在https://github.com/joyent/node/tree/v0.4.8這里看到源碼。

js2c.py

node.js使用了V8附帶的js2c.py工具把所有內置的js代碼轉換成C++里的數組,生成node_natives.h直接include到程序中,成了C++源碼的一部分。這樣做能提高內置js模塊的編譯效率。

node.js里內置的javascript包括了主程序src/node.js和模塊程序lib/*.js,通過js2c.py讓每一個js文件都生成一個源碼數組,存放在build/src/node_natives.h里,node_natives.h在node.js編譯后才會生成(編譯的腳本wscript中調用了js2c.py),可以看到大致的結構如下:

  1. namespace node {  
  2. const char node_native[] = {47, 47, 32, 67, 112 ......}  
  3. const char console_native[] = {47, 47, 32, 67, 112 ......}  
  4. const char buffer_native[] = {47, 47, 32, 67, 112 ......}  
  5. .....  
  6. }  
  7. struct _native { const char* name; const char* source; size_t source_len;};  
  8. static const struct _native natives[] = {  
  9. "node", node_native, sizeof(node_native)-1 },  
  10. "dgram", dgram_native, sizeof(dgram_native)-1 },  
  11. "console", console_native, sizeof(console_native)-1 },  
  12. "buffer", buffer_native, sizeof(buffer_native)-1 },  
  13. ....  

這個文件被包含在node_javascript.cc里,node_javascript.cc提供了兩個接口:

MainSource() 處理node_native源碼返回v8::Handle類型的數據可供編譯。

DefineJavaScript(target) 把其他所有模塊源碼變成v8::Handle類型后加載到傳入的target對象上。

所有的js模塊都被轉換成了C數組,接下來看看它們怎么執行和互相調用。

執行js主程序/傳遞process

先看看node.js的底層C++傳遞給javascript的一個變量process,在一開始運行node.js時,程序會先配置好process

  1. Handleprocess = SetupProcessObject(argc, argv); 

然后把process作為參數去調用js主程序src/node.js返回的函數,這樣process就傳遞到javascript里了。

  1. //node.cc  
  2. //通過MainSource()獲取已轉化的src/node.js源碼,并執行它  
  3. Local f_value = ExecuteString(MainSource(), IMMUTABLE_STRING("node.js"));  
  4.  
  5. //執行src/node.js后獲得的是一個函數,從node.js源碼可以看出:  
  6. //node.js  
  7. //(function(process) {  
  8. // global = this;  
  9. // ....  
  10. //})  
  11. Local f = Local::Cast(f_value);  
  12.  
  13.  
  14. //創建函數執行環境,調用函數,把process傳入  
  15. Localglobal = v8::Context::GetCurrent()->Global();  
  16. Local args[1] = { Local::New(process) };  
  17. f->Call(global, 1, args); 

C++模塊

node.js的模塊除了lib/*.js里用js語言編寫的以外,還有一些使用C++編寫,像os/stdio/crypto/buffer等。這些模塊都通過node.h提供的NODE_MODULE方法存儲在變量_module里。node_extensions.cc提供了get_builtin_module(name)接口獲取這些模塊。

process.binding/C++模塊加載

process提供的一個獲取模塊的接口是binding,它的實現Binding()函數可以在node.cc找到。

  1. Persistent binding_cache;  
  2. static Handle Binding(const Arguments& args) {  
  3. HandleScope scope;  
  4. Local module = args[0]->ToString();  
  5. String::Utf8Value module_v(module);  
  6. node_module_struct* modp;  
  7.  
  8. if (binding_cache.IsEmpty()) {  
  9. binding_cache = Persistent::New(Object::New());  
  10. }  
  11. Local exports;  
  12. if (binding_cache->Has(module)) {  
  13. exports = binding_cache->Get(module)->ToObject();  
  14.  
  15. else if ((modp = get_builtin_module(*module_v)) != NULL) {  
  16. exports = Object::New();  
  17. modp->register_func(exports);  
  18. binding_cache->Set(module, exports);  
  19.  
  20. else if (!strcmp(*module_v, "constants")) {  
  21. exports = Object::New();  
  22. DefineConstants(exports);  
  23. binding_cache->Set(module, exports);  
  24.  
  25. #ifdef __POSIX__  
  26. else if (!strcmp(*module_v, "io_watcher")) {  
  27. exports = Object::New();  
  28. IOWatcher::Initialize(exports);  
  29. binding_cache->Set(module, exports);  
  30. #endif  
  31.  
  32. else if (!strcmp(*module_v, "natives")) {  
  33. exports = Object::New();  
  34. DefineJavaScript(exports);  
  35. binding_cache->Set(module, exports);  
  36.  
  37.  
  38. else {  
  39. return ThrowException(Exception::Error(String::New("No such module")));  
  40. }  
  41. return scope.Close(exports);  

從源碼可以看到,調用process.binding時,先看緩存里是否已經存在此模塊,不存在再調用get_builtin_module查找C++內置模塊,找到的話獲取后綁定在exports上,在最后返回exports。

此外還有針對其他模塊的特殊處理,其中natives模塊就是調用上文提到的DefineJavaScript(exports)接口獲取到所有內置的js模塊綁定在exports上。

現在在js上需要調用C++提供的模塊只需要調用process.binding就行了,例如

  1. var stdio = process.binding("stdio"

js模塊加載

src/node.js上實現了一個NativeModule對象用于管理js模塊,它通過調用process.binding(“natives”)把所有內置的js模塊放在NativeModule._source上,并提供require接口供調用。在require里會給代碼加一層包裝,把一些變量傳給這個模塊。

  1. NativeModule.wrapper = [  
  2. '(function (exports, require, module, __filename, __dirname) { ',  
  3. '\n});' 
  4. ]; 

再用process提供的其中一個js編譯接口process.runInThisContext執行代碼。

  1. var fn = runInThisContext(source, this.filename, true);  
  2. fn(this.exports, NativeModule.require, thisthis.filename); 

于是在主程序src/node.js上可以調用NativeModule.require(“net”)去加載net模塊,在lib/*.js的各個js模塊里能通過調用傳進來的require()去加載其他內置js模塊。

總結流程

粗略總結一下加載模塊的流程:

加載C++模塊(以stdio為例):

process.binding(“stdio”) -> get_builtin_module(“stdio”) -> _module -> NODE_MODULE(node_stdio, node::Stdio::Initialize)(定義)

加載js模塊(以net為例)

require(“net”) -> NativeModule.require(“net”) -> process.binding(“natives”)["net"] -> DefineJavaScript() -> natives[] -> node_natives.h

原文:http://cnodejs.org/blog/?p=1280

【編輯推薦】

  1. 揭秘Node.js事件
  2. Node.js初探之與Mysql的交互
  3. Node.js初探之hello world
  4. Node.js入門之神秘的服務器端JavaScript
  5. 什么是Node.js?
責任編輯:陳貽新 來源: cnodejs.org
相關推薦

2019-12-17 11:40:44

Node.js模塊前端

2020-08-31 15:00:17

Node.jsrequire前端

2021-09-26 05:06:04

Node.js模塊機制

2025-05-26 00:31:31

2020-04-15 15:48:03

Node.jsstream前端

2011-09-08 10:41:12

Node.js

2019-12-10 10:23:57

Node.jsCluster前端

2013-11-01 09:34:56

Node.js技術

2015-03-10 10:59:18

Node.js開發指南基礎介紹

2023-06-30 23:25:46

HTTP模塊內存

2011-12-09 11:16:48

Node.js

2011-11-10 08:55:00

Node.js

2012-10-24 14:56:30

IBMdw

2011-09-08 13:46:14

node.js

2011-11-01 10:30:36

Node.js

2011-09-02 14:47:48

Node

2011-09-09 14:23:13

Node.js

2011-09-08 10:32:27

Node.js

2021-12-25 22:29:57

Node.js 微任務處理事件循環

2023-06-20 19:35:00

Node.js工具
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 人人做人人澡人人爽欧美 | 免费观看一区二区三区毛片 | 成年无码av片在线 | 精品国模一区二区三区欧美 | 免费毛片在线 | 久久91| 国产美女一区二区三区 | 国产精品免费一区二区三区四区 | 亚洲日本激情 | 日韩精品一区二区三区 | 欧美激情精品久久久久久免费 | 黄片毛片免费看 | 久久99精品久久久久久国产越南 | 国产精品一区二区三区在线 | 美女二区 | 丁香久久 | 亚洲免费在线 | 亚洲欧洲综合av | 精品一区二区在线观看 | 色综合久 | 亚洲精品久久久一区二区三区 | 午夜不卡一区二区 | 91精品国产麻豆 | 91亚洲精品在线观看 | av电影一区| 国产亚洲一区二区三区在线 | 青青草原综合久久大伊人精品 | 久久精品国产一区 | 日本特黄a级高清免费大片 特黄色一级毛片 | 欧美精品欧美精品系列 | avav在线看| 午夜国产羞羞视频免费网站 | 亚洲导航深夜福利涩涩屋 | 国产二区视频 | 国产一区二区三区久久 | 午夜免费av| 97热在线 | 亚洲国产一区二区三区四区 | av一级久久 | 99久热 | 欧美在线观看一区 |