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

news2025/1/12 16:03:42

在 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++ 进行绑定。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1342696.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

中介者模式-Mediator Pattern-1

如果在一个系统中对象之间的联系呈现为网状结构&#xff0c; 对象之间存在大量的多对多联系&#xff0c;将导致系统非常复杂。 这些对象既会影响别的对象&#xff0c;也会被别的对象所影响。 这些对象称为同事对象&#xff0c;它们之间通过彼此的相互作用实现系统的行为。 在网…

如何本地搭建FastDFS文件服务器并实现远程访问【内网穿透】

文章目录 前言1. 本地搭建FastDFS文件系统1.1 环境安装1.2 安装libfastcommon1.3 安装FastDFS1.4 配置Tracker1.5 配置Storage1.6 测试上传下载1.7 与Nginx整合1.8 安装Nginx1.9 配置Nginx 2. 局域网测试访问FastDFS3. 安装cpolar内网穿透4. 配置公网访问地址5. 固定公网地址5.…

2024年软考中级-网络工程师

网络工程师证书考到后&#xff0c;通过本级考试的合格人员能根据应用部门的要求进行网络系统的规划、设计和网络设备的软硬件安装调试工作&#xff0c;能进行网络系统的运行、维护和管理&#xff0c;能高效、可靠、安全地管理网络资源&#xff1b;作为网络专业人员对系统开发进…

sqlite3 c++ VS编译生成静态库

官网 https://www.sqlite.org/download.html 下载sqlite-amalgamation和x86版本下载sqlite-dll-win32-x86、x64位版本sqlite-dll-win64-x64 解压 SQLITE-AMALGAMATION包含 shell.csqlite3.csqlite3.hsqlite3ext.hsqlite-dll-win32-x86包含 sqlite3.def sqlite3.dll建立一个空…

使用ArcMap进行选址服务,适宜性分析

文章目录 题目分析技术步骤1&#xff0c;环境设置2&#xff0c;计算坡度&#xff1a;空间分析——表面分析——坡度&#xff0c;根据DEM求坡度4&#xff0c;距离计算3&#xff0c;坡度赋分4&#xff0c;对学校赋分5&#xff0c;娱乐设施赋分6&#xff0c;土地利用类型赋分7&…

springboot整合hadoop遇错

错误一&#xff1a; Caused by: java.io.FileNotFoundException: HADOOP_HOME and hadoop.home.dir are unset. 解决&#xff1a; 下载&#xff1a;https://github.com/steveloughran/winutils 选择一个版本 例如&#xff1a;3.0.0 &#xff0c;将里面的hadoop.dll文件复制…

Linux磁盘阵列

一.RAID磁盘阵列介绍 RAID&#xff08;Redundatnt Array of lndependent Disks&#xff09;&#xff0c;全称为&#xff1a;独立冗余磁盘阵列 解释&#xff1a; RAID是一种把多块独立的硬盘&#xff08;物理硬盘&#xff09;按不同的方式组合起来形成一个硬盘组&#xff08;逻…

【教程】自动检测和安装Python脚本依赖的第三方库

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhang.cn] 背景说明 对于新python环境&#xff0c;要运行某个脚本&#xff0c;可能需要安装很多库&#xff0c;一般可以通过提供的requirements.txt来自动安装。但如果没有这个txt&#xff0c;那就得手动一个一个安装&#…

Live800:客服售前、售中,售后需要做哪些服务?有哪些区别?

客服在售前、售中和售后阶段扮演着重要的角色&#xff0c;他们为顾客提供全方位的服务&#xff0c;确保顾客的满意度和忠诚度。尽管这三个阶段都涉及到与顾客的互动&#xff0c;但每个阶段的服务内容和重点有所不同。 在售前阶段&#xff0c;客服的主要任务是提供产品或服务的信…

深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第三节 栈与堆,值类型与引用类型

深入浅出图解C#堆与栈 C# Heaping VS Stacking 第三节 栈与堆&#xff0c;值类型与引用类型 [深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第一节 理解堆与栈](https://mp.csdn.net/mdeditor/101021023)[深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第二节 栈基本工…

视频监控EasyCVR如何通过设置sei接口,实现在webrtc视频流中添加画框和文字?

安防视频监控系统基于视频综合管理平台EasyCVR视频系统&#xff0c;采用了开放式的网络结构&#xff0c;可以提供实时远程视频监控、视频录像、录像回放与存储、告警、语音对讲、云台控制、平台级联、磁盘阵列存储、视频集中存储、云存储等丰富的视频能力&#xff0c;具备权限管…

炎症状态重塑了急性髓系白血病的免疫微环境并改善了风险分层

今天给同学们分享一篇实验文章“An inflammatory state remodels the immune microenvironment and improves risk stratification in acute myeloid leukemia”&#xff0c;这篇文章发表在Nat Cancer期刊上&#xff0c;影响因子为22.7。 结果解读&#xff1a; 成人和儿童AML的…

掌握比特币生态话语权,或将成为下一轮牛市爆发的掌舵人!

伴随着美国SEC批准比特币现货的强烈的市场预期&#xff0c;比特币财富效应日益显著&#xff0c;比特币价格更是从年初的16500美金一路上涨到了44000美金&#xff0c;而代币上涨带来的大量资金的涌入&#xff0c;也促使比特币生态也再次进入了高速发展期&#xff0c;铭文资产在比…

**Python**综合案例

Python综合案例 一、系统需求分析 1、需求分析 使用面向对象编程思想完成学员管理系统的开发,具体如下: ① 系统要求:学员数据存储在文件中 ② 系统功能:添加学员、删除学员、修改学员信息、查询学员信息、显示所有学员信息、保存学员信息及退出系统等功能。 2、角色…

C/C++ 函数参数按引用传递、指针传递、实参传递

引用经常被用作函数参数&#xff0c;使得函数中的变量名成为调用程序中的变量的别名。这种传递参数的方法称为按引用传递。按引用传递允许被调用的函数能够访问调用函数中的变量。C新增的这项特性是对 C语言的超越&#xff0c;C语言只能按值传递。按值传递导致被调用函数使用调…

【JS 逆向百例】steam 登录 Protobuf 协议详解

声明 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;不提供完整代码&#xff0c;抓包内容、敏感网址、数据接口等均已做脱敏处理&#xff0c;严禁用于商业用途和非法用途&#xff0c;否则由此产生的一切后果均与作者无关&#xff01; 本文章未…

Java创建线程执行任务的方法(一)

目录 1.继承Thread类 2.实现Runnab类 2.1实现Runnable类 2.2使用Lambda表达式 3.实现Callable类 3.1返回Integer类型数据 3.2返回String类型数据 3.3返回Object类型数据 4.匿名内部类 创建线程的方法&#xff1a;继承Thread类&#xff1b;实现Runnab类&#xff1b;匿名…

java使用JSON工具解析字符串、数组详解

一&#xff1a;问题 1.最近自己在前后端数据交互时需要进行JSON格式字符串、数组数据进行转换&#xff0c;进行问题整理 2.遇到需要JSON字符串转换的朋友可以阅读 二&#xff1a;解析步骤 1.第一点首先确定需求&#xff0c;明确需要转的字符串是一个对象还是一个数组&#…

腾讯云服务器怎么购买?流程来了

腾讯云服务器购买流程直接在官方秒杀活动上购买比较划算&#xff0c;在云服务器CVM或轻量应用服务器页面自定义购买价格比较贵&#xff0c;但是自定义购买云服务器CPU内存带宽配置选择范围广&#xff0c;活动上购买只能选择固定的活动机&#xff0c;选择范围窄&#xff0c;但是…

lag-llama源码解读(Lag-Llama: Towards Foundation Models for Time Series Forecasting)

Lag-Llama: Towards Foundation Models for Time Series Forecasting 文章内容&#xff1a; 时间序列预测任务&#xff0c;单变量预测单变量&#xff0c;基于Llama大模型&#xff0c;在zero-shot场景下模型表现优异。创新点&#xff0c;引入滞后特征作为协变量来进行预测。 获得…