Node.js 和 C++ 之间的类型转换

我非常喜欢使用 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 类型的变量。

#include <node.h>#include <v8.h>

using namespace v8;

void Test(const v8::FunctionCallbackInfo<v8::Value>& args) {    Isolate* isolate = args.GetIsolate();    // 声明变量    Local<Number> retval = v8::Number::New(isolate, 1000);}

void init(Local <Object> exports, Local<Object> module) {    NODE_SET_METHOD(exports, "getTestValue", Test);}

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。

#include <node.h>#include <v8.h>

using namespace v8;

void MyFunction(const v8::FunctionCallbackInfo<Value>& args) {    Isolate* isolate = args.GetIsolate();    args.GetReturnValue().Set(String::NewFromUtf8(isolate, "Hello World!"));}

void Test(const v8::FunctionCallbackInfo<v8::Value>& args) {    Isolate* isolate = args.GetIsolate();

// Number 类型的声明    Local<Number> retval = v8::Number::New(isolate, 1000);

// String 类型的声明    Local<String> str = v8::String::NewFromUtf8(isolate, "Hello World!");

// Object 类型的声明    Local<Object> obj = v8::Object::New(isolate);    // 对象的赋值    obj->Set(v8::String::NewFromUtf8(isolate, "arg1"), str);    obj->Set(v8::String::NewFromUtf8(isolate, "arg2"), retval);

// Function 类型的声明并赋值    Local<FunctionTemplate> tpl = v8::FunctionTemplate::New(isolate, MyFunction);    Local<Function> fn = tpl->GetFunction();    // 函数名字    fn->SetName(String::NewFromUtf8(isolate, "theFunction"));    obj->Set(v8::String::NewFromUtf8(isolate, "arg3"), fn);

// Boolean 类型的声明    Local<Boolean> flag = Boolean::New(isolate, true);    obj->Set(String::NewFromUtf8(isolate, "arg4"), flag);

// Array 类型的声明    Local<Array> arr = Array::New(isolate);    // Array 赋值    arr->Set(0, Number::New(isolate, 1));    arr->Set(1, Number::New(isolate, 10));    arr->Set(2, Number::New(isolate, 100));    arr->Set(3, Number::New(isolate, 1000));    obj->Set(String::NewFromUtf8(isolate, "arg5"), arr);

// Undefined 类型的声明    Local<Value> und = Undefined(isolate);    obj->Set(String::NewFromUtf8(isolate, "arg6"), und);

// null 类型的声明    Local<Value> null = Null(isolate);    obj->Set(String::NewFromUtf8(isolate, "arg7"), null);

// 返回给 JavaScript 调用时的返回值    args.GetReturnValue().Set(obj);}

void init(Local <Object> exports, Local<Object> module) {    NODE_SET_METHOD(exports, "getTestValue", Test);}

NODE_MODULE(returnValue, init)

所有的 addon 都需要一个初始化的函数,如下面的代码:

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

Initialize 是初始化的函数,module_name 是编译后产生的二进制文件名,上述代码的模块名为 returnValue

上述代码通过 node-gyp 编译后(编译过程官方文档 C/C++ Addons 有详细的介绍),可以通过如下的方式调用。

// returnValue.node 这个文件就是编译后产生的文件,通过 NODE_MODULE(returnValue, init) 决定的文件名const returnValue = require('./build/Release/returnValue.node');console.log(returnValue.getTestValue());

运行结果如下:

数据流向 javaScript -> C++

上面的 demo 展示了怎样在在 C++ 定义 JavaScript 类型,数据的是从 C++ 流向 JavaScript,反过来数据也需要从 javaScript 流向 C++,也就是调用 C++ 函数的时候需要传入一些参数。

下面的代码展示了参数个数判断,参数类型判断,以及参数类型装换成 V8 类型的过程,包括基本类型以及 Object, Array, Fuction。

#include <node.h>#include <v8.h>#include <iostream>

using namespace v8;using namespace std;

void GetArgument(const FunctionCallbackInfo<Value>& args) {    Isolate* isolate = args.GetIsolate();

// 参数长度判断    if (args.Length() < 2) {        isolate->ThrowException(Exception::TypeError(            String::NewFromUtf8(isolate, "Wrong number of arguments")));        return;    }

// 参数类型判断    if (!args[0]->IsNumber() || !args[1]->IsNumber()) {        //抛出错误        isolate->ThrowException(Exception::TypeError(            String::NewFromUtf8(isolate, "argumnets must be number")));    }

if (!args[0]->IsObject()) {        printf("I am not Object\n");    }

if (!args[0]->IsBoolean()) {        printf("I am not Boolean\n");    }

if (!args[0]->IsArray()) {        printf("I am not Array\n");    }

if (!args[0]->IsString()) {        printf("I am not String\n");    }

if (!args[0]->IsFunction()) {        printf("I am not Function\n");    }

if (!args[0]->IsNull()) {        printf("I am not Null\n");    }

if (!args[0]->IsUndefined()) {        printf("I am not Undefined\n");    }

// js Number 类型转换成 v8 Number 类型    Local<Number> value1 = Local<Number>::Cast(args[0]);    Local<Number> value2 = Local<Number>::Cast(args[1]);    double value = value1->NumberValue() + value2->NumberValue();

// js String 类型转换成 v8 String 类型    Local<String> str = Local<String>::Cast(args[2]);    String::Utf8Value utfValue(str);    cout<<string(*utfValue)<<endl;

// js Array 类型转换成 v8 Array 类型    Local<Array> input_array = Local<Array>::Cast(args[3]);    printf("%d, %f %f\n", input_array->Length(), input_array->Get(0)->NumberValue(), input_array->Get(1)->NumberValue());

// js Object 类型转换成 v8 Object 类型    Local<Object> obj = Local<Object>::Cast(args[4]);

// 根据 key 获取对象中的值    Local<Value> a = obj->Get(String::NewFromUtf8(isolate, "a"));    Local<Value> b = obj->Get(String::NewFromUtf8(isolate, "b"));

// js Array 类型转换成 v8 Array 类型    Local<Array> c = Local<Array>::Cast(obj->Get(String::NewFromUtf8(isolate, "c")));    cout<<a->NumberValue()<<"   "<<b->NumberValue()<<endl;    printf("%d, %f %f\n", c->Length(), c->Get(0)->NumberValue(), c->Get(1)->NumberValue());

// js String 类型转换成 v8 String 类型    Local<String> cString = Local<String>::Cast(c->Get(2));    String::Utf8Value utfValueD(cString);    cout<<string(*utfValueD)<<endl;

// 根据 key 获取对象中的值    Local<Object> d = Local<Object>::Cast(obj->Get(String::NewFromUtf8(isolate, "d")));    Local<String> dString1 = Local<String>::Cast(d->Get(String::NewFromUtf8(isolate, "m")));    String::Utf8Value utfValued1(dString1);    cout<<string(*utfValued1)<<endl;

// 根据 key 获取对象中的值    Local<String> dString2 = Local<String>::Cast(d->Get(String::NewFromUtf8(isolate, "n")));    String::Utf8Value utfValued2(dString2);    cout<<string(*utfValued2)<<endl;

// js Booelan 类型转换成 v8 Boolean 类型    Local<Boolean> FlagTrue = Local<Boolean>::Cast(args[5]);    cout<<"Flag: "<<FlagTrue->BooleanValue()<<endl;

// js Function 类型转换成 v8 Function 类型    Local<Function> cb = Local<Function>::Cast(args[8]);    const unsigned argc = 2;    Local<Value> argv[2];    argv[0] = a;    argv[1] = b;    cb->Call(Null(isolate), argc, argv);

args.GetReturnValue().Set(value);}

void Init(Local <Object> exports, Local <Object> module) {    NODE_SET_METHOD(module, "exports", GetArgument);}

NODE_MODULE(argumentss, Init)

通过 node-gyp 编译后,可以通过如下的方式调用。

const getArguments = require('./build/Release/arguments');

console.log(getArguments(2, 3, 'Hello Arguments', [1, 2, 3], {        a: 10,        b: 100,        c: [23, 22, "我是33"],        d: { m: '我是22', n: '我是23' }    }, true, null, undefined,    function myFunction(...args) {        console.log('I am Function!');        console.log(...args);        console.log('I am Function!');    }));

运行结果如下:

关于其他的类型,我这里就就不一一介绍,V8 文档里面都有对应的 API。

NAN

由于 V8 的 API 还没有彻底稳定下来,所以对于不同版本的 Node.js 类型相关的 API 会发生变化,而 NAN 帮我们做了封装,在编码的时候不需要关心版本问题,只需要引入相应的头文件即可。

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

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

参考资料

转载自:http://taobaofed.org/blog/2016/09/20/type-casts-between-node-and-cpp/

作者:慎里

时间: 2024-10-03 13:22:54

Node.js 和 C++ 之间的类型转换的相关文章

在 Node.js 和 C++ 之间使用 Buffer 共享数据

本文讲的是在 Node.js 和 C++ 之间使用 Buffer 共享数据, 使用 Node.js 开发的一个好处是简直能够在 JavaScript 和 原生 C++ 代码之间无缝切换 - 这要得益于 V8 的扩展 API.从 JavaScript 进入 C++ 的能力有时由处理速度驱动,但更多的情况是我们已经有 C++ 代码,而我们想要直接用 JavaScript 调用. 我们可以用(至少)两轴对不同用例的扩展进行分类 - (1)C++ 代码的运行时间,(2)C++ 和 JavaScript

Node.js与PHP之间通过RSA算法实现签名验证

Node.js与PHP之间通过RSA算法实现签名验证 做过web开发的同学都知道我们经常在接口间交互的时候为了防止数据伪造并 且保证安全性我们会采用签名的方式进行验证, 签名的算法我们一般选择RSA非对称加密算法.想了解更多签名与RSA算法知识可以去查阅PKCS#1.PKCS#7相关资料. 笔者在做利用Node.js做单点登录的时候涉及到了Node.js项目中产生的签名,然后需要到PHP项目中去验签,具体实现如下. 首先准备RSA公钥和私钥,格式均为PEM openssl产生pem格式私钥 op

[译] 在你沉迷于包的海洋之前,还是了解一下运行时 Node.js 的本身

本文讲的是[译] 在你沉迷于包的海洋之前,还是了解一下运行时 Node.js 的本身, 原文地址:Before you bury yourself in packages, learn the Node.js runtime itself 原文作者:该文章已获得作者 Samer Buna 授权 译文出自:掘金翻译计划 译者:fghpdf 校对者:rccoder,reid3290 在你沉迷于包的海洋之前,还是了解一下运行时 Node.js 的本身 这篇文章将挑战你 Node.js 的知识极限. 我

提升 Node.js 应用性能的 5 个技巧

"如果nginx没有在你的节点服务器之前,那么你可能就错了."Bryan Hughes在Twitter上说 Node.js是全球领先的用JavaScript--世界上最流行的编程语言创建服务器应用程序的工具.提供web服务器和应用服务器的功能,Node.js被认为是各种以微服务为基础的开发和交付的关键工具. (下载关于Node.js和NGINX的免费Forrester报告.) Node.js可以替代或增强Java和.NET用于后端应用程序的开发. Node.js是单线程的,并且使用非阻

《写给PHP开发者的Node.js学习指南》一第 2 章 简单的Node.js框架2.1 HTTP服务器

第 2 章 简单的Node.js框架 写给PHP开发者的Node.js学习指南 在之前的章节,我介绍了一个用于PHP到Node.js转换的开发环境,以及如何使用它进行转换.在本章,我们将开始使用这个开发环境并进行实际的转换. 2.1 HTTP服务器 写给PHP开发者的Node.js学习指南 在PHP中,一个PHP文件代表一个HTML页面.一个Web服务器,比如Apache,当请求一个PHP页面时,Web服务器会运行PHP.但是在Node.js里,Node.js的main文件代表了整个服务器.No

PHP vs Node.js 深入讨论

网络正在处于一个日新月异的发展时代.服务器端开发人员在选择语言的时候非常困惑,有长期占主导地位的语言,例如C.Java和Perl,也有专注于web开发的语言,例如Ruby.Clojure和Go.只要你的项目运行良好,你的选择就显得没有那么重要了. 但是如何让这些新的web开发人员做出一个正确的选择呢? 我不希望展开一场PHP.NodeJs两大阵营之间的战争,我将会对比这两种语言所在领域的发展状况: PHP Rasmus Lerdorf在1994年创造出了PHP.它是由安装在web服务器(Apac

NodeJS和C++之间的类型转换

我非常喜欢使用 Node.js,但是当涉及到计算密集型的场景时 Node.js 就不能够很好地胜任了.而在这样的情况下 C++ 是一个很好的选择,非常幸运 Node.js 官方提供了C/C++ Addons 的机制让我们能够使用 V8 API 把 Node.js 和 C++ 结合起来. 虽然在 Node.js 官方网站有很多的关于怎么使用这些 API 的文档,但是在 JavaScript 和 C++ 之间传递数据是一件非常麻烦的事情,C++ 是强类型语言("1024" 是字符串类型而不

Node.js 服务端实践之 GraphQL 初探

0.问题来了 DT 时代,各种业务依赖强大的基础数据平台快速生长,如何高效地为各种业务提供数据支持,是所有人关心的问题. 现有的业务场景一般是这样的,业务方提出需求,然后寻找开发资源,由后端提供数据,让前端实现各种不同的业务视图.这样的做法存在很多的重复劳动,如果能够将其中通用的内容抽取出来提供给各个业务方反复使用,必然能够节省宝贵的开发时间和开发人力. 前端的解决方案是将视图组件化,各个业务线既可以是组件的使用者,也可以是组件的生产者.那么问题来了,前端通过组件实现了跨业务的复用,后端接口如何

PHP 和 Node.js 的10项对比挑战

在最近 SitePoint 的 PHP vs Node.js Smackdown 一文中,Craig Buckler 对两种语言就如何应对一系列的10个挑战进行了比较来决定哪一个总体上更佳. Craig 在书中讲到,这些比较总是有些矛盾.作为一个有意思的随访,我们要求 Bruno Škvorc (SitePoint 的 PHP 开发者)和 James Hibbard (SitePoint 的一个 JavaScript 开发者)对每一轮提供评论. 下面是他们详细的看法- 第一轮:开始 Round