节点.js和 C/C++ 集成:如何正确实现回调

Node.js and C/C++ integration: how to properly implement callbacks?

本文关键字:何正确 实现 回调 集成 C++ js 节点      更新时间:2023-10-16

我正在尝试实现一个C++扩展以与node.js集成。此扩展将在内部调用一些阻塞调用,因此它需要为节点.js世界提供一个非阻塞接口。

如 https://nodejs.org/api/addons.html 中所述,有两种方法可以实现非阻塞回调:

a) 通过使用对 JavaScript 函数的简单回调。所以我的扩展必须生成一个线程并立即返回,并让该线程调用阻塞代码,然后在返回时调用 JavaScript 回调。这似乎相对容易实现。

b) 通过使用 libuv 库,如果我理解正确的话,将事件发布到 node.js 事件循环。我没有详细阅读 libuv 文档,但这似乎很难实现。

我的偏好当然是a),但我不知道这意味着什么。如果从不同的线程调用回调,从而产生节点.js非阻塞 IO 的标准方法,是否有任何问题?还是需要使用 libuv 来正确处理我的代码及其阻塞调用的线程?

非常感谢您的帮助。

从不同的线程调用回调不是一个选项,v8 不允许这样做。所以你必须选择b。实际上实施起来并不难。我建议使用 nan 来完成此任务。下面是文档中的示例:

class PiWorker : public NanAsyncWorker {
 public:
  PiWorker(NanCallback *callback, int points)
    : NanAsyncWorker(callback), points(points) {}
  ~PiWorker() {}
  // Executed inside the worker-thread.
  // It is not safe to access V8, or V8 data structures
  // here, so everything we need for input and output
  // should go on `this`.
  void Execute () {
    estimate = Estimate(points);
  }
  // Executed when the async work is complete
  // this function will be run inside the main event loop
  // so it is safe to use V8 again
  void HandleOKCallback () {
    NanScope();
    Local<Value> argv[] = {
        NanNull()
      , NanNew<Number>(estimate)
    };
    callback->Call(2, argv);
  };
 private:
  int points;
  double estimate;
};
// Asynchronous access to the `Estimate()` function
NAN_METHOD(CalculateAsync) {
  NanScope();
  int points = args[0]->Uint32Value();
  NanCallback *callback = new NanCallback(args[1].As<Function>());
  NanAsyncQueueWorker(new PiWorker(callback, points));
  NanReturnUndefined();
}

在后台,它将处理使用 libuv 线程池调用您的代码并在主线程上调用回调。