国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

NodeJS和C++之間的類型轉換

開發 后端
雖然在 Node.js 官方網站有很多的關于怎么使用這些 API 的文檔,但是在 JavaScript 和 C++ 之間傳遞數據是一件非常麻煩的事情,C++ 是強類型語言(”1024” 是字符串類型而不是整數類型),而 JavaScript 卻總是默認的幫我們做一些類型轉換。

我非常喜歡使用 Node.js,但是當涉及到計算密集型的場景時 Node.js 就不能夠很好地勝任了。而在這樣的情況下 C++ 是一個很好的選擇,非常幸運 Node.js 官方提供了C/C++ Addons 的機制讓我們能夠使用 V8 API 把 Node.js 和 C++ 結合起來。

雖然在 Node.js 官方網站有很多的關于怎么使用這些 API 的文檔,但是在 JavaScript 和 C++ 之間傳遞數據是一件非常麻煩的事情,C++ 是強類型語言(”1024” 是字符串類型而不是整數類型),而 JavaScript 卻總是默認的幫我們做一些類型轉換。

JavaScript 的基本類型包括 String,Number,Boolean,null,undefined,V8 使用類繼承的方式來定義這類型,這些類型都繼承了 Primitive 類,而 Primitive 繼承了 Value,v8 也支持整型(包括 Int32 和 Uint32),而所有的類型定義都可以從 V8 類型文檔中看到,除了基本的類型,還有 Object,Array,Map 等類型的定義。

基本類型的繼承關系如下圖:

在 V8 中所有 JavaScript 值都是被放在 Local 對象中,通過這個對象指定了 JavaScript 運行時的內存單元。

下面這段代定義了一個 Number 類型的值,其中 Test 函數中聲明的 isolate 變量代表著 V8 虛擬機中的堆內存,當創建新變量的時候就需要用到它,接下來的一行代碼就通過 isolate 聲明了一個 Number 類型的變量。

 

  1. #include <node.h> 
  2. #include <v8.h> 
  3.  
  4. using namespace v8; 
  5.  
  6. void Test(const v8::FunctionCallbackInfo<v8::Value>& args) { 
  7.     Isolate* isolate = args.GetIsolate(); 
  8.     // 聲明變量 
  9.     Local<Number> retval = v8::Number::New(isolate, 1000); 
  10.  
  11. void init(Local <Object> exports, Local<Object> module) { 
  12.     NODE_SET_METHOD(exports, "getTestValue", Test); 
  13.  
  14. NODE_MODULE(returnValue, init) 

 

看了 V8 類型 API 文檔 你會發現對于基本的 JavaScript 類型,只有變量的聲明而沒有變量的賦值。最初想可能覺得這個非常的奇怪,可是仔細想一想后發現這個是合理的。主要由以下幾點原因:

  • JavaScript 的基本類型是不可變類型,變量都是指向一個不可變的內存單元,var a = 10,則 a 指向的內存單元中包含的值為 5,重新賦值 a = 100,沒有改變這個內存單元的值,而是使得 a 指向了另外一個內存單元,其中的值為 100。如果聲明兩個變量 x,y 的值都為 10,則他們指向的是同一個內存單元。

  • 函數的傳參都是傳值,而不是傳引用,當在 JavaScript 中調用 C++ 的函數時,如果參數是基本類型則每次都是把這個值拷貝過去,改變參數的值不會影響原來的值。

  • 使用 Local<Value> 聲明基本類型的變量都是對內存單元的引用,因為***條原因不可能改變引用的值使其指向另外一個內存單元,因此不存在變量的重新賦值。

數據流向 C++ -> JavaScript

下面 demo 定義了一些常用的 JavaScript 類型,包括基本類型的以及 Object, Array, Fuction。

 

  1. #include <node.h> 
  2. #include <v8.h> 
  3.  
  4. using namespace v8; 
  5.  
  6. void MyFunction(const v8::FunctionCallbackInfo<Value>& args) { 
  7.     Isolate* isolate = args.GetIsolate(); 
  8.     args.GetReturnValue().Set(String::NewFromUtf8(isolate, "Hello World!")); 
  9.  
  10. void Test(const v8::FunctionCallbackInfo<v8::Value>& args) { 
  11.     Isolate* isolate = args.GetIsolate(); 
  12.  
  13.     // Number 類型的聲明 
  14.     Local<Number> retval = v8::Number::New(isolate, 1000); 
  15.  
  16.     // String 類型的聲明 
  17.     Local<String> str = v8::String::NewFromUtf8(isolate, "Hello World!"); 
  18.  
  19.     // Object 類型的聲明 
  20.     Local<Object> obj = v8::Object::New(isolate); 
  21.     // 對象的賦值 
  22.     obj->Set(v8::String::NewFromUtf8(isolate, "arg1"), str); 
  23.     obj->Set(v8::String::NewFromUtf8(isolate, "arg2"), retval); 
  24.  
  25.     // Function 類型的聲明并賦值 
  26.     Local<FunctionTemplate> tpl = v8::FunctionTemplate::New(isolate, MyFunction); 
  27.     Local<Function> fn = tpl->GetFunction(); 
  28.     // 函數名字 
  29.     fn->SetName(String::NewFromUtf8(isolate, "theFunction")); 
  30.     obj->Set(v8::String::NewFromUtf8(isolate, "arg3"), fn); 
  31.  
  32.     // Boolean 類型的聲明 
  33.     Local<Boolean> flag = Boolean::New(isolate, true); 
  34.     obj->Set(String::NewFromUtf8(isolate, "arg4"), flag); 
  35.  
  36.     // Array 類型的聲明 
  37.     Local<Array> arr = Array::New(isolate); 
  38.     // Array 賦值 
  39.     arr->Set(0, Number::New(isolate, 1)); 
  40.     arr->Set(1, Number::New(isolate, 10)); 
  41.     arr->Set(2, Number::New(isolate, 100)); 
  42.     arr->Set(3, Number::New(isolate, 1000)); 
  43.     obj->Set(String::NewFromUtf8(isolate, "arg5"), arr); 
  44.  
  45.     // Undefined 類型的聲明 
  46.     Local<Value> und = Undefined(isolate); 
  47.     obj->Set(String::NewFromUtf8(isolate, "arg6"), und); 
  48.  
  49.     // null 類型的聲明 
  50.     Local<Value> null = Null(isolate); 
  51.     obj->Set(String::NewFromUtf8(isolate, "arg7"), null); 
  52.  
  53.     // 返回給 JavaScript 調用時的返回值 
  54.     args.GetReturnValue().Set(obj); 
  55.  
  56. void init(Local <Object> exports, Local<Object> module) { 
  57.     NODE_SET_METHOD(exports, "getTestValue", Test); 
  58.  
  59. NODE_MODULE(returnValue, init) 

 

所有的 addon 都需要一個初始化的函數,如下面的代碼:

  1. void Initialize(Local<Object> exports); 
  2. NODE_MODULE(module_name, Initialize) 

Initialize 是初始化的函數,module_name 是編譯后產生的二進制文件名,上述代碼的模塊名為returnValue

上述代碼通過 node-gyp 編譯后(編譯過程官方文檔 C/C++ Addons 有詳細的介紹),可以通過如下的方式調用。

  1. // returnValue.node 這個文件就是編譯后產生的文件,通過 NODE_MODULE(returnValue, init) 決定的文件名 
  2. const returnValue = require('./build/Release/returnValue.node'); 
  3. console.log(returnValue.getTestValue()); 

運行結果如下:

數據流向 javaScript -> C++

上面的 demo 展示了怎樣在在 C++ 定義 JavaScript 類型,數據的是從 C++ 流向 JavaScript,反過來數據也需要從 javaScript 流向 C++,也就是調用 C++ 函數的時候需要傳入一些參數。

下面的代碼展示了參數個數判斷,參數類型判斷,以及參數類型裝換成 V8 類型的過程,包括基本類型以及 Object, Array, Fuction。

 

  1. #include <node.h> 
  2. #include <v8.h> 
  3. #include <iostream> 
  4.  
  5. using namespace v8; 
  6. using namespace std; 
  7.  
  8. void GetArgument(const FunctionCallbackInfo<Value>& args) { 
  9.     Isolate* isolate = args.GetIsolate(); 
  10.  
  11.     // 參數長度判斷 
  12.     if (args.Length() < 2) { 
  13.         isolate->ThrowException(Exception::TypeError( 
  14.             String::NewFromUtf8(isolate, "Wrong number of arguments"))); 
  15.         return
  16.     } 
  17.  
  18.     // 參數類型判斷 
  19.     if (!args[0]->IsNumber() || !args[1]->IsNumber()) { 
  20.         //拋出錯誤 
  21.         isolate->ThrowException(Exception::TypeError( 
  22.             String::NewFromUtf8(isolate, "argumnets must be number"))); 
  23.     } 
  24.  
  25.     if (!args[0]->IsObject()) { 
  26.         printf("I am not Object\n"); 
  27.     } 
  28.  
  29.     if (!args[0]->IsBoolean()) { 
  30.         printf("I am not Boolean\n"); 
  31.     } 
  32.  
  33.     if (!args[0]->IsArray()) { 
  34.         printf("I am not Array\n"); 
  35.     } 
  36.  
  37.     if (!args[0]->IsString()) { 
  38.         printf("I am not String\n"); 
  39.     } 
  40.  
  41.     if (!args[0]->IsFunction()) { 
  42.         printf("I am not Function\n"); 
  43.     } 
  44.  
  45.     if (!args[0]->IsNull()) { 
  46.         printf("I am not Null\n"); 
  47.     } 
  48.  
  49.     if (!args[0]->IsUndefined()) { 
  50.         printf("I am not Undefined\n"); 
  51.     } 
  52.  
  53.     // js Number 類型轉換成 v8 Number 類型 
  54.     Local<Number> value1 = Local<Number>::Cast(args[0]); 
  55.     Local<Number> value2 = Local<Number>::Cast(args[1]); 
  56.     double value = value1->NumberValue() + value2->NumberValue(); 
  57.  
  58.     // js String 類型轉換成 v8 String 類型 
  59.     Local<String> str = Local<String>::Cast(args[2]); 
  60.     String::Utf8Value utfValue(str); 
  61.     cout<<string(*utfValue)<<endl; 
  62.  
  63.     // js Array 類型轉換成 v8 Array 類型 
  64.     Local<Array> input_array = Local<Array>::Cast(args[3]); 
  65.     printf("%d, %f %f\n", input_array->Length(), input_array->Get(0)->NumberValue(), input_array->Get(1)->NumberValue()); 
  66.  
  67.     // js Object 類型轉換成 v8 Object 類型 
  68.     Local<Object> obj = Local<Object>::Cast(args[4]); 
  69.  
  70.     // 根據 key 獲取對象中的值 
  71.     Local<Value> a = obj->Get(String::NewFromUtf8(isolate, "a")); 
  72.     Local<Value> b = obj->Get(String::NewFromUtf8(isolate, "b")); 
  73.  
  74.     // js Array 類型轉換成 v8 Array 類型 
  75.     Local<Array> c = Local<Array>::Cast(obj->Get(String::NewFromUtf8(isolate, "c"))); 
  76.     cout<<a->NumberValue()<<"   "<<b->NumberValue()<<endl; 
  77.     printf("%d, %f %f\n", c->Length(), c->Get(0)->NumberValue(), c->Get(1)->NumberValue()); 
  78.  
  79.     // js String 類型轉換成 v8 String 類型 
  80.     Local<String> cString = Local<String>::Cast(c->Get(2)); 
  81.     String::Utf8Value utfValueD(cString); 
  82.     cout<<string(*utfValueD)<<endl; 
  83.  
  84.     // 根據 key 獲取對象中的值 
  85.     Local<Object> d = Local<Object>::Cast(obj->Get(String::NewFromUtf8(isolate, "d"))); 
  86.     Local<String> dString1 = Local<String>::Cast(d->Get(String::NewFromUtf8(isolate, "m"))); 
  87.     String::Utf8Value utfValued1(dString1); 
  88.     cout<<string(*utfValued1)<<endl; 
  89.  
  90.     // 根據 key 獲取對象中的值 
  91.     Local<String> dString2 = Local<String>::Cast(d->Get(String::NewFromUtf8(isolate, "n"))); 
  92.     String::Utf8Value utfValued2(dString2); 
  93.     cout<<string(*utfValued2)<<endl; 
  94.  
  95.     // js Booelan 類型轉換成 v8 Boolean 類型 
  96.     Local<Boolean> FlagTrue = Local<Boolean>::Cast(args[5]); 
  97.     cout<<"Flag: "<<FlagTrue->BooleanValue()<<endl; 
  98.  
  99.     // js Function 類型轉換成 v8 Function 類型 
  100.     Local<Function> cb = Local<Function>::Cast(args[8]); 
  101.     const unsigned argc = 2
  102.     Local<Value> argv[2]; 
  103.     argv[0] = a; 
  104.     argv[1] = b; 
  105.     cb->Call(Null(isolate), argc, argv); 
  106.  
  107.     args.GetReturnValue().Set(value); 
  108.  
  109. void Init(Local <Object> exports, Local <Object> module) { 
  110.     NODE_SET_METHOD(module, "exports", GetArgument); 
  111.  
  112. NODE_MODULE(argumentss, Init) 

 

運行結果如下:

[[178264]]

關于其他的類型,我這里就就不一一介紹,V8 文檔里面都有對應的 API。

NAN

由于 V8 的 API 還沒有徹底穩定下來,所以對于不同版本的 Node.js 類型相關的 API 會發生變化,而 NAN 幫我們做了封裝,在編碼的時候不需要關心版本問題,只需要引入相應的頭文件即可。

引入頭文件后,可以如下使用方式:

v8::Local<v8::Primitive> Nan::Undefined()
v8::Local<v8::Primitive> Nan::Null()

參考資料

 

責任編輯:張燕妮 來源: 慎里
相關推薦

2016-12-20 16:35:52

NodeJSC++類型轉換

2009-08-28 10:08:02

C#數值類型之間轉換

2023-08-14 08:35:36

2010-01-28 15:22:12

C++嵌套類

2011-07-13 18:00:51

CC++VC

2010-02-03 11:11:18

C++類型轉換

2011-07-14 10:39:08

強制類型轉換函數C++

2010-01-28 13:45:06

C++數組

2020-01-09 09:50:32

C++JavaPython

2023-09-04 11:00:54

CC++語言

2024-02-28 09:57:30

C++類型強制轉換開發

2010-01-25 15:55:50

托管C++

2010-01-28 16:31:54

C++類型

2024-05-21 14:04:16

2025-01-10 06:30:00

2009-08-12 18:16:47

C#類型比較

2011-06-08 13:50:39

C#類型轉換

2009-08-20 11:29:21

C#16進制字符串

2010-01-20 09:54:27

C++數據類型

2025-05-20 08:10:00

函數函數類型函數指針類型
點贊
收藏

51CTO技術棧公眾號

先锋影音成人资源| 欧洲精品在线播放| v片在线观看| 久久伊99综合婷婷久久伊| 国产精品久久久久久久av大片| 污污网站在线看| 亚洲理论在线观看| 日韩精品综合在线| 国产日韩1区| 欧美一级淫片aaaaaaa视频| 国产激情小视频在线| 亚欧色一区w666天堂| 十八禁视频网站在线观看| 蜜桃视频在线一区| 国产精品推荐精品| av一区二区高清| 久久人人看视频| 国产福利91精品一区二区| 91精品国产aⅴ一区二区| 色琪琪免费视频网站| 久久久另类综合| 在线国产99| 老牛嫩草一区二区三区日本| 成人性生交大片免费看视频直播 | 大地资源第二页在线观看高清版| 欧美午夜精彩| 欧美精品成人91久久久久久久| 在线观看特色大片免费视频| 精品视频免费在线| 你懂的视频在线观看| 亚洲免费观看视频| 青青草精品视频在线观看| 99免费精品视频| 免费cad大片在线观看| 捆绑紧缚一区二区三区视频| 久久综合久久综合这里只有精品| 国内精品久久久久久久影视麻豆| 国产精品中文久久久久久久| 欧美精品一区二区三区精品| 欧美在线视频在线播放完整版免费观看| 伊人久久噜噜噜躁狠狠躁| www.xxxx精品| 精品久久久久久久久久岛国gif| 精品国产美女在线| 国产在线视频欧美一区| 精品国内亚洲在观看18黄| 四虎视频在线精品免费网址| 欧美成人精品一区二区三区| 亚洲国产精品免费视频| 国内精品久久久久久中文字幕| 成人中文字幕视频| 国产精品美女免费| 91超碰成人| 玛丽玛丽电影原版免费观看1977| 国产精品一区毛片| 一区二区三区四区欧美| 国产精品一二三四| 国产av人人夜夜澡人人爽麻豆| 99在线视频精品| 蜜臀av免费观看| 亚洲成人av资源| 午夜视频在线观看网站| 日韩美女在线视频| 国产综合色区在线观看| 欧美插天视频在线播放| 欧美黄色录像| 亚洲www在线观看| 香蕉久久夜色精品| 男人的天堂avav| 国产精品久久久久桃色tv| 免费看美女隐私的视频| 欧美怡红院视频| 人成在线免费网站| 久久久久久97| 欧美午夜不卡| 香港三级日本三级a视频| 日本一区二区三区dvd视频在线| 97影视大全免费追剧大全在线观看| 欧美亚洲动漫精品| 在线日韩影院| 日韩av手机在线| 男人的天堂亚洲| 欧美精品第三页| 在线看不卡av| julia一区二区三区中文字幕| 国内外成人免费激情在线视频| 欧美激情麻豆| 国产69精品久久久久久久| 亚洲综合免费观看高清在线观看| 欧美精品电影| 欧美乱妇高清无乱码| 欧美日韩国产色综合一二三四| 永久免费看av| 狠狠躁天天躁日日躁欧美| 亚洲美女炮图| 国产精品久久久久久久久久免费| 校园激情久久| 自拍偷拍一区二区三区四区| 欧美日免费三级在线| 欧美久久一区二区三区| 精品国产免费人成电影在线观...| 99热99精品| 日本韩国在线视频爽| 欧美激情高清视频| 日韩中文字幕亚洲一区二区va在线 | 国产福利精品在线| 日av在线不卡| 诱人的瑜伽老师3hd中字| 精品对白一区国产伦| 日本午夜精品| 日韩欧美国产一二三区| 亚洲妇熟xx妇色黄| aaa欧美色吧激情视频| 欧美日韩第二页| 春暖花开亚洲| 精品中文字幕在线观看| 日韩精品视频在线观看网址| 韩国女主播一区二区三区| 色999五月色| 亚洲一区二区成人在线观看| 在线免费三级电影网站| 91精品视频播放| 国产精品视频九色porn| 亚洲一区资源| 精品国产乱码一区二区三区四区 | 中文在线√天堂| 欧美老少做受xxxx高潮| 另类人妖一区二区av| 国产原创av在线| 国产97人人超碰caoprom| 26uuu国产电影一区二区| 白浆在线视频| 欧美主播一区二区三区美女 久久精品人 | 91精彩视频在线播放| 欧美在线免费看| 国产午夜亚洲精品羞羞网站| av中文字幕电影在线看| 国产精品一区二区免费看| 亚洲自拍偷拍综合| 精品亚洲自拍| 国产成人av影视| 日韩最新av在线| 国产成人自拍网| mm视频在线视频| 国产一区二区三区四区五区在线 | 性史性dvd影片农村毛片| 久久久久久中文字幕| 99视频一区二区| 成人在线不卡| 男的插女的下面视频| 亚洲女人天堂av| 国产在线精品一区二区三区不卡| 欧美色图天堂| 手机成人在线| 精品国产髙清在线看国产毛片| 午夜一区二区三区不卡视频| √新版天堂资源在线资源| 亚洲一区制服诱惑| 日本韩国欧美一区二区三区| 亚洲蜜桃视频| av福利精品| 日本不卡在线播放| 日韩国产中文字幕| 国产91精品在线观看| 国产精品久久久久久久久免费高清| 久操手机在线视频| 免费91在线视频| 国产精品久久久久久久久久久免费看 | 极品尤物一区二区三区| 欧美三级午夜理伦三级中视频| 亚洲精品欧美| 成人三级小说| 黄色激情在线视频| 欧美日韩成人在线视频| 国产精品欧美久久久久无广告| 午夜精品影视国产一区在线麻豆| 88av在线| 国产一区免费| 日韩精品欧美国产精品忘忧草| 国产成人av一区二区三区在线| 国产情侣一区在线| 午夜3点看的视频| 精品卡一卡二| 亚洲老司机av| 久久这里都是精品| 成人高清av| 久久综合网导航| 国产一区二区三区小说| 8x海外华人永久免费日韩内陆视频| 精品久久久久久久久久| 国产精品综合| 青草综合视频| 制服黑丝国产在线| 日韩黄色影视| 久久天堂av综合合色| 亚洲国产另类精品专区| 久久精品伊人| 精品国产乱码一区二区三区 | 亚洲色图网站| 久久男人av资源站|