探索 Node.js 与 C++ 的绑定:使用 node-addon-api

2023-12-30 13:28

本文主要是介绍探索 Node.js 与 C++ 的绑定:使用 node-addon-api,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在 Node.js 中使用 C++ 进行绑定是一种强大的方式,可以充分利用 C++ 的性能优势。在本文中,我们将探讨如何使用 node-addon-api 来实现这一目标。

1. 为什么选择 C++ 绑定?

Node.js 是一个基于 JavaScript 的平台,它使得开发人员能够使用 JavaScript 进行高性能的网络应用开发。然而,有时我们可能需要更高的性能,这时我们可以考虑使用 C++。通过将关键部分用 C++ 编写并绑定到 Node.js 中,我们可以获得更好的性能。

2. node-addon-api 简介

node-addon-api 是一个为 Node.js 编写的 C++ 插件提供的高级 C++ API。它提供了一组功能强大的工具,使得开发人员能够轻松地在 C++ 和 JavaScript 之间进行交互。

3. 安装和设置

3.1 安装 Node.js 和 npm:

# 安装 Node.js  
wget https://nodejs.org/dist/v16.14.2/node-v16.14.2-linux-x64.tar.xz  
tar xvf node-v16.14.2-linux-x64.tar.xz  
ln -s /path/to/node-v16.14.2-linux-x64/bin/* /usr/local/bin/  # 验证安装  
node -v  
npm -v

3.2 创建新项目

mkdir my-node-addon  
cd my-node-addon

3.3 初始化项目

npm init -y

3.4 安装 node-addon-api:

node-gyp configure build

3.5 在 Node.js 中使用插件

编译完成后,你可以在 Node.js 中使用你的插件。以下是一个简单的示例:

// 根据实际情况调整路径。 
const addon = require('./build/Release/addon');
// 输出 2。注意,这只是一个简单的示例,实际情况可能会更复杂。你可能需要处理错误、进行类型检查等。
console.log(addon.add(1)); 

4. 编写 C++ 插件

4.1 创建 C++ 源文件

在 src 目录下创建一个新文件,例如 addon.cc

4.2 编写 C++ 代码

在 addon.cc 文件中,使用 node-addon-api 的 API 来编写你的 C++ 代码。例如:

#include <node_addon_api.h>  napi_value Add(napi_env env, napi_callback_info info) {  size_t argc = 1;  napi_value args[1];  NAPI_CALL(env, napi_get_cb_info(env, info, &argc, args, nullptr, nullptr));  int32_t result = 0;  if (argc >= 1) {  NAPI_CALL(env, napi_get_value_int32(env, args[0], &result));  }  return napi_value_wrap_int32(env, result + 1); // 这里只是简单地将输入参数加一并返回。  
}

4.3 暴露 API

在 binding.gyp 文件中定义你的 API:

{  "targets": [  {  "target_name": "myaddon",  "sources": ["addon.cc"],  "include_dirs": ["<!(node -p \"require('node-addon-api').include\")"]  }  ]  
}

然后在 src 目录下创建一个 binding.gyp 文件,并将上面的代码复制到该文件中。这将告诉 node-gyp 如何编译你的插件。 

4.4 编译插件

在项目根目录中运行以下命令:

node-gyp configure build --target=v16.14.2 --arch=x64 --build_type=Release --msvs_version=2019 --dist_summary=full --dist_summary_format=full --no-rebuild --force-process-config --force-clean --verbose --napi_version=4 --napi_build_version=0 --napi_nodejs_version=v16.14.2 --napi_build_type=Release --napi_build64=false --napi_legacy_base_node=false --napi_compiler=clang++ --napi_deployment_target=89 --napi_default_libraries=false --napi_multiversion=false --napi_parent_path=src --napi_buildroot=out --napi_libroot=out/Release/obj.target --napi_sharedlinkflags="" --napi_sharedlibs="" --napi_sharedlibsonlyflags="" --napi_sharedlibslinkflags="" --napi_sharedlibslibs="" --napi_nodejsroot="" --napi_distfile="" --napi_installroot="" --napi_nodejslibname="" --napi_build32=true --napi_build64=false --napi_nodejslib32=out/Release/obj.target/nodejs/defaultlib.target/src/nodejslib.node --napi_nodejslib64=out/Release/obj.target/nodejs/defaultlib.target/obj/defaultlib.o ""myaddon"" "--modulemap=out/Release/obj.target/nodejs/defaultlib.target/src/myaddon/myaddon.modulemap"" "--backendflags=--no-tsan"" "--backendflags=""" "--backendflags

4.5 测试和调试

使用 Node.js 运行你的插件。例如,创建一个名为 test.js 的文件,并编写以下代码:

const addon = require('./build/Release/myaddon');  
console.log(addon.add(2, 3)); // 调用你定义的 Add 函数,并传入两个参数 2 和 3

在项目根目录中运行以下命令:

node test.js

 这将运行你的测试代码并输出结果。如果一切正常,你应该看到输出 5

5. 在 Node.js 中使用插件

5.1 引入插件

在你的 JavaScript 文件中,使用 require 函数来引入编译后的插件文件。例如:

const addon = require('./build/Release/myaddon');

5.2 调用 C++ 函数

使用引入的插件对象来调用你在 C++ 中定义的函数。例如:

const result = addon.add(1, 2);  
console.log(result); // 输出 3,这是 C++ 函数 Add 的返回值

5.3 处理错误和异常

在调用 C++ 函数时,确保妥善处理可能出现的错误和异常。使用 try-catch 语句来捕获并处理异常。例如:

try {  const result = addon.add(1, 2);  console.log(result); // 输出 3,这是 C++ 函数 Add 的返回值  
} catch (error) {  console.error('An error occurred:', error);  
}

5.4 内存管理

由于 Node.js 使用 V8 引擎,因此需要特别注意内存管理。避免在 C++ 中直接操作 JavaScript 的对象,以防止出现内存泄漏或错误的数据类型转换。在 C++ 中,你应该使用 napi_create_* 系列函数来创建和操作 JavaScript 对象。例如:

在 C++ 中:

napi_value CreateObject(napi_env env, napi_callback_info info) {  napi_value obj;  NAPI_CALL(env, napi_create_object(env, &obj));  return obj;  
}

在 JavaScript 中:

const addon = require('./build/Release/myaddon');  
// 调用 C++ 函数来创建一个 JavaScript 对象并将其返回给 JavaScript 代码。
const obj = addon.CreateObject(); 

 6. 在 JavaScript 中处理回调函数和 Promise

在使用插件时,你可能需要调用返回 Promise 的函数或使用回调函数。处理这种情况的一种常见方式是使用 async/await 语法。以下是一个示例:

async function myFunction() {  try {  const result = await addon.myPromiseFunction(); // 调用返回 Promise 的 C++ 函数  console.log(result); // 输出 Promise 的结果  } catch (error) {  console.error('An error occurred:', error);  }  
}

在这个例子中,myFunction 是一个异步函数,它使用 await 关键字等待 addon.myPromiseFunction() 的结果。如果 Promise 被拒绝,则抛出异常并被 catch 语句捕获。

如果你需要传递回调函数给 C++ 插件,可以使用 napi_create_function 创建一个 JavaScript 函数,并将其作为参数传递给 C++ 函数。例如:

在 JavaScript 中:

const callback = async (result) => {  console.log('Callback called with result:', result);  
};  addon.myCallbackFunction(callback); // 调用 C++ 函数并传递回调函数作为参数

 在 C++ 中:

void MyCallbackFunction(napi_env env, napi_callback_info info) {  napi_value result;  // ... 从其他地方获取 result ...  napi_value callback;  NAPI_CALL(env, napi_get_cb_info(env, info, nullptr, nullptr, &callback));  NAPI_CALL(env, napi_call_function(env, callback, result)); // 调用回调函数并传递结果作为参数  
}

请注意,上述示例中的回调函数是一个异步函数,并且使用 await 关键字等待 C++ 函数的返回结果。你需要确保正确处理任何可能抛出的异常。 

结论

通过使用 node-addon-api,我们可以轻松地在 Node.js 中使用 C++ 进行绑定。这为我们提供了一种强大的方式,可以在 Node.js 中利用 C++ 的性能优势。然而,需要注意错误处理、类型转换、内存管理、性能优化和测试等方面的问题。如果你能够妥善处理这些问题,那么你就可以成功地在 Node.js 中使用 C++ 进行绑定。

这篇关于探索 Node.js 与 C++ 的绑定:使用 node-addon-api的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/552972

相关文章

JS常用组件收集

收集了一些平时遇到的前端比较优秀的组件,方便以后开发的时候查找!!! 函数工具: Lodash 页面固定: stickUp、jQuery.Pin 轮播: unslider、swiper 开关: switch 复选框: icheck 气泡: grumble 隐藏元素: Headroom

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

Hadoop数据压缩使用介绍

一、压缩原则 (1)运算密集型的Job,少用压缩 (2)IO密集型的Job,多用压缩 二、压缩算法比较 三、压缩位置选择 四、压缩参数配置 1)为了支持多种压缩/解压缩算法,Hadoop引入了编码/解码器 2)要在Hadoop中启用压缩,可以配置如下参数

Makefile简明使用教程

文章目录 规则makefile文件的基本语法:加在命令前的特殊符号:.PHONY伪目标: Makefilev1 直观写法v2 加上中间过程v3 伪目标v4 变量 make 选项-f-n-C Make 是一种流行的构建工具,常用于将源代码转换成可执行文件或者其他形式的输出文件(如库文件、文档等)。Make 可以自动化地执行编译、链接等一系列操作。 规则 makefile文件

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�