将任意Javascript数据对象传递给Node.js C ++插件

时间:2017-09-14 15:21:41

标签: javascript c++ json node.js node.js-nan

我有一个使用Nan用C ++编写的Node.js插件。工作非常好。但是,我无法弄清楚如何让我的Node Javascript代码将任意数据对象(例如{attr1:42, attr2:'hi', attr3:[5,4,3,2,1]})传递给C ++插件。

到目前为止,我通过在我的数据对象上调用JSON.stringify()然后在C ++端解析字符串化的JSON来解决这个问题。

理想情况下,我想避免复制数据,只是获取对我可以访问的数据对象的引用,或者至少本地复制它并避免字符串化/解析......

任何帮助将不胜感激!

2 个答案:

答案 0 :(得分:7)

您可以允许Node.js c ++插件采用任意类型的参数,但必须明确检查和处理类型。他是一个简单的示例函数,展示了如何执行此操作:

void args(const Nan::FunctionCallbackInfo<v8::Value>& info) {

    int i = 0;
    while (i < info.Length()) {
        if (info[i]->IsBoolean()) {
            printf("boolean = %s", info[i]->BooleanValue() ? "true" : "false");
        } else if (info[i]->IsInt32()) {
            printf("int32 = %ld", info[i]->IntegerValue());
        } else if (info[i]->IsNumber()) {
            printf("number = %f", info[i]->NumberValue());
        } else if (info[i]->IsString()) {
            printf("string = %s", *v8::String::Utf8Value(info[i]->ToString()));
        } else if (info[i]->IsObject()) {
            printf("[object]");
            v8::Local<v8::Object> obj = info[i]->ToObject();
            v8::Local<v8::Array> props = obj->GetPropertyNames();
            for (unsigned int j = 0; j < props->Length(); j++) {
                printf("%s: %s",
                       *v8::String::Utf8Value(props->Get(j)->ToString()),
                       *v8::String::Utf8Value(obj->Get(props->Get(j))->ToString())
                      );
            }
        } else if (info[i]->IsUndefined()) {
            printf("[undefined]");
        } else if (info[i]->IsNull()) {
            printf("[null]");
        }
        i += 1;
    }
}

为了实际解决处理可能包含具有任意数据的对象的任意参数的问题,我建议编写一个解析实际对象的函数,类似于我在此示例中解析函数参数的方式。请记住,如果您希望能够处理对象中的嵌套对象,则可能需要递归执行此操作。

答案 1 :(得分:3)

您不必将对象字符串化以将其传递给c ++插件。有接受这些的方法 仲裁对象。但它不是那么随意。您必须编写不同的代码来解析c ++中的对象。 将其视为数据库的模式。您无法在单个集合/表中保存不同的格式数据。 您将需要具有特定架构的另一个表/集合。

让我们看看这个例子:

我们将对象{x:10,y:5}传递给addon,c ++ addon将返回另一个对象,其中包含sum和product 像这样的属性:{x1:15,y1:50}

在cpp代码中:

NAN_METHOD(func1) {
        if (info.Length() > 0) {
                Local<Object> obj = info[0]->ToObject();
                Local<String> x = Nan::New<String>("x").ToLocalChecked();
                Local<String> y = Nan::New<String>("y").ToLocalChecked();

                Local<String> sum  = Nan::New<String>("sum").ToLocalChecked();
                Local<String> prod  = Nan::New<String>("prod").ToLocalChecked();

                Local<Object> ret = Nan::New<Object>();

                double x1 = Nan::Get(obj, x).ToLocalChecked()->NumberValue();
                double y1 = Nan::Get(obj, y).ToLocalChecked()->NumberValue();

                Nan::Set(ret, sum, Nan::New<Number>(x1 + y1));
                Nan::Set(ret, prod, Nan::New<Number>(x1 * y1));

                info.GetReturnValue().Set(ret);

        }
}

在javascript ::

const addon = require('./build/Release/addon.node');
var obj = addon.func1({ 'x': 5, 'y': 10 });
console.log(obj); // { sum: 15, prod: 50 }

在这里,您只能将{x: (Number), y: (number)}类型对象仅发送到插件。否则它将无法解析或 检索数据。

对于数组来说就像这样:

在cpp:

NAN_METHOD(func2) {
    Local<Array> array = Local<Array>::Cast(info[0]);

    Local<String> ss_prop = Nan::New<String>("sum_of_squares").ToLocalChecked();
    Local<Array> squares = New<v8::Array>(array->Length());
    double ss = 0;

    for (unsigned int i = 0; i < array->Length(); i++ ) {
      if (Nan::Has(array, i).FromJust()) {
        // get data from a particular index
        double value = Nan::Get(array, i).ToLocalChecked()->NumberValue();

        // set a particular index - note the array parameter
        // is mutable
        Nan::Set(array, i, Nan::New<Number>(value + 1));
        Nan::Set(squares, i, Nan::New<Number>(value * value));
        ss += value*value;
      }
    }
    // set a non index property on the returned array.
    Nan::Set(squares, ss_prop, Nan::New<Number>(ss));
    info.GetReturnValue().Set(squares);
}

在javascript中:

const addon = require('./build/Release/addon.node');
var arr = [1, 2, 3];
console.log(addon.func2(arr));  //[ 1, 4, 9, sum_of_squares: 14 ]

像这样,您可以处理数据类型。如果你想要复杂的对象或操作,你就是这样 必须在一个函数中混合使用这些方法并解析数据。