C++ 从节点模拟 int main(int argc, char *argv[])

C++ Emulating int main(int argc, char *argv[]) from node

基本上,我正在使用他们的 NAPI 功能将现有的 C++ 文件与我们的 javascript / 节点集成。我让它在测试 C++ 文件上工作,所以我知道我的设置工作正常。但是,在实际的 C++ 文件中,它被设计为从命令行使用 argc 和 argv 从命令行 运行。基本上我只需要从我的其他函数内部调用 C++ 中的 main 方法,这意味着没有命令行。所以我必须传入 argc 和 argv 的值。 argc 只是一个 int,这很简单,但是 argc 是一个 char ** 类型,根据我的研究,它看起来像是一个字符数组数组又名字符串?

这是我的 C++ 文件底部的当前代码

void Init(Env env, Object exports, Object module) {
    exports.Set("main", Function::New(env, main(2,{"test","test2"})));
}
NODE_API_MODULE(addon, Init)

argc 值工作正常

我正在尝试为 argv 创建一个要传入的临时/测试值,但我在弄清楚如何使用我的值创建 char ** 类型的数组时遇到了问题。

argv是一个字符串指针数组(实际上是NUL结尾的字符数组),其中元素0是程序名,元素1 ... argc-1是程序参数,元素 argc 必须是 NULL1.

C++ 中没有数组文字,因此您必须显式创建数组变量以将其传递给函数。更糟糕的是,允许 main 修改传递的参数,因此您甚至无法构建字符串文字数组,因为它们是只读的;因此,您必须为每个参数显式分配 read/write space。准系统解决方案可以是为每个参数设置单个缓冲区并从中构建指针数组:

char argv0[] = "test_program";
char argv1[] = "arg1";
char argv2[] = "arg2";
char *argv[] = {argv0, argv1, argv2, NULL};
main(3, argv);

一个更灵活的方法(特别是如果你必须动态构建你的参数)可以使用 std::vector<std::string>:

std::vector<std::string> args = { "test_program", "arg1", "arg2" };
// ... here you may add other arguments dynamically...
args.push_back("arg3"); // whatever
// build the pointers array
std::vector<char *> argv;
for(std::string &s: args) argv.push_back(&s[0]);
argv.push_back(NULL);
main(argv.size()-1, argv.data());

现在,进入您的代码:

void Init(Env env, Object exports, Object module) {
    exports.Set("main", Function::New(env, main(2,{"test","test2"})));
}
NODE_API_MODULE(addon, Init)

除了您无法构建 argv 以传递给 main 的事实之外,您还试图 调用 main 并传递它的结果作为 Function::New 的第二个参数,而 Function::New wants some callable type(例如函数指针)注册为名为 main 的导出的处理程序!从 Function::New 文档复制:

/// Callable must implement operator() accepting a const CallbackInfo&
/// and return either void or Value.

因此,作为一个简单的示例,您可以将 main 导出为无参数的 JS 函数,returns 什么都没有(undefined,我猜?)通过注册这样的回调:

void MainCallback(const CallbackInfo& info) {
    char argv0[] = "test_program";
    char argv1[] = "arg1";
    char argv2[] = "arg2";
    char *argv[] = {argv0, argv1, argv2, NULL};
    main(3, argv);
}

void Init(Env env, Object exports, Object module) {
    exports.Set("main", Function::New(env, MainCallback));
}
NODE_API_MODULE(addon, Init)

最后,正如其他人所说,从技术上讲,在 C++ 中 main 有点神奇——从程序内部调用它是未定义的行为;实际上,在我所知道的任何平台上也可以 运行 node.js,main 是一个完全正常的函数,恰好在启动时被 C 运行time 调用,所以我认为这不会给您带来任何问题。


备注

  1. 因此,您可以说它是以 NULL 结尾的字符数组组成的以 NULL 结尾的数组。注意这里 NULL = 空指针; NUL = 字符串终止符,即 '[=32=]'.