Nodejs C++插件(N-API)

N-API 作为Nodejs项目的一部分,旨在更加便捷稳定的创建Nodejs Native 插件。这个API作为应用二进制接口(ABI),在Node.js的各个版本中都是稳定的,而且ABI允许一个在主要版本编译的模块,在以后的Node.js版本上运行,而无需重新编译。

0. 环境搭建

  1. 安装最新的npm,且Nodejs版本10.x以上
  2. 工具链
    • Windows执行 npm install --global windows-build-tools
    • MacOS执行 xcode-select --install
    • Linux使用系统自带的GCC即可
  3. 安装构建工具 npm install --global node-gyp
  4. 初始化测试项目 npm init
  5. 添加napi模块npm i node-addon-api
  6. 为VSCode添加智能提示所需头文件(以Windows为例)
    • Ctrl+Shift+P -> C++ 添加 c_cpp_properties.json 文件

    • 执行 node-gyp configure 查看 -Dnode_root_dir 选项打印,确认node头文件目录

    • c_cpp_properties.json includePath 中加入 C:\\Users\\gaobowen\\AppData\\Local\\node-gyp\\Cache\\10.16.0\\include\\node

1. JS中调用C++方法

1.1 JS中调用源文件的C++方法

  1. 编写功能函数

    test.h、test.cc
#ifndef SAMPLE_TEST_H
#define SAMPLE_TEST_H
#include <napi.h>
#include <windows.h>
#include <sstream>
#include <thread> Napi::Value plus(const Napi::CallbackInfo& info); #endif
#include "test.h"

Napi::Value plus(const Napi::CallbackInfo& info) {
Napi::Env env = info.Env();
Napi::EscapableHandleScope scope(env);
if (info.Length() < 1)
return Napi::Boolean::New(env, false);
if (!info[0].IsNumber())
return Napi::Boolean::New(env, false);
if (!info[1].IsNumber())
return Napi::Boolean::New(env, false);
auto a = info[0].As<Napi::Number>().DoubleValue();
auto b = info[1].As<Napi::Number>().DoubleValue(); return Napi::Number::New(env, a + b);
}
  1. 编写导出函数

    exports.cc
#include <napi.h>
#include "test.h" Napi::Object Init(Napi::Env env, Napi::Object exports)
{
exports.Set(Napi::String::New(env, "plus"), Napi::Function::New(env, plus));
return exports;
} NODE_API_MODULE(napi_sample, Init)
  1. 添加 binding.gyp 文件
{
"targets": [
{
"target_name": "napi_sample",
"sources": [ ],
"cflags!": [ "-fno-exceptions" ],
"cflags_cc!": [ "-fno-exceptions" ],
"defines": [ "NAPI_DISABLE_CPP_EXCEPTIONS" ],
"include_dirs": [
"<!--@(node -p \"require('node-addon-api').include\")"
],
"conditions": [
['OS=="win"', {
"sources": [ "sample/exports.cc", "sample/test.cc" ],
}]
]
}
]
}
  1. 命令行执行 node-gyp rebuild 生成 ./build/Release/napi_sample.node文件
  2. 测试使用napi_sample.node插件

    node .\test.js
let { plus } = require('./build/Release/napi_sample.node');

console.log('test export=-->', plus);
console.log('test call=>', plus(1.2, 1.3));

控制台输出

test export=> function () { [native code] }
test call=> 2.5

1.2 JS中调用动态库的C++方法

Napi::Value call_dll(const Napi::CallbackInfo& info) {
Napi::Env env = info.Env();
Napi::EscapableHandleScope scope(env);
//注,若动态库还有其他依赖库,请放在同一级目录下
auto m = LoadLibraryExA("user32.dll", NULL, LOAD_WITH_ALTERED_SEARCH_PATH);
typedef int(*MyMessageBoxA)(int, char*, char*, int);
auto box = (MyMessageBoxA)GetProcAddress(m, "MessageBoxA");
box(0, "js call dll function", "abc", 0); return Napi::Boolean::New(env, true);
}

控制台输出

test call_dll=> true

2. C++中调用JS方法

2.1. C++单线程调用JS方法

test.cpp

Napi::Value cpp_call_js(const Napi::CallbackInfo& info) {
Napi::Env env = info.Env();
Napi::EscapableHandleScope scope(env);
if (info.Length() < 0)
return Napi::Boolean::New(env, false);
if (!info[0].IsFunction())
return Napi::Boolean::New(env, false);
auto jscb = info[0].As<napi::function>(); //回调函数入参
napi_value a, b;
napi_create_double(env, 1, &a);
napi_create_double(env, 2, &b); napi_value argv[] = { a, b }; //回调函数返回值
napi_value result;
napi_call_function(env, env.Global(), jscb, 2, argv, &result); return Napi::Value(env, result);
}

test.js

let { cpp_call_js } = require('./build/Release/napi_sample.node');
(async function () { let c = 3;
let js_callback = function(a, b){
return a + b + c;
}
console.log('test cpp_call_js=>', cpp_call_js(js_callback)); })();

控制台输出

test cpp_call_js=> 6

2.2. C++多线程调用JS方法

test.cpp

Napi::ThreadSafeFunction tsfunc;
void thread_function() {
auto threadid = std::this_thread::get_id();
std::stringstream ss;
ss << threadid;
auto threadidstr = new std::string(ss.str()); auto callback = [threadid](Napi::Env env, Napi::Function jscb, std::string* p_str) {
jscb.Call({ Napi::String::New(env, *p_str) });
delete p_str;
}; tsfunc.Acquire();
napi_status status = tsfunc.BlockingCall(threadidstr, callback);
if (status != napi_ok) {
printf("tsfunc.BlockingCall error.\n");
}
tsfunc.Release(); } Napi::Value cpp_thread_call_js(const Napi::CallbackInfo& info) {
Napi::Env env = info.Env();
Napi::EscapableHandleScope scope(env);
if (info.Length() < 0)
return Napi::Boolean::New(env, false);
if (!info[0].IsFunction())
return Napi::Boolean::New(env, false);
auto jscb = info[0].As<napi::function>();
tsfunc = Napi::ThreadSafeFunction::New(env, jscb, "my-tsfunc", 2, 2);
std::thread(thread_function).join();
std::thread(thread_function).join();
return Napi::Boolean::New(env, true);
}

test.js

let { cpp_thread_call_js } = require('./build/Release/napi_sample.node');
(async function () { let js_callback_async = function(id){
console.log('thread_id',id);
}
console.log('test cpp_thread_call_js=>', cpp_thread_call_js(js_callback_async)); })();

控制台输出

test cpp_thread_call_js=> true
thread_id 6768
thread_id 12352

DEMO地址

Nodejs C++插件(N-API)的更多相关文章

  1. Linux中安装nodejs及插件

    Linux中安装nodejs及插件 1.去官网下载安装包 英文网址:https://nodejs.org/en/download/ 中文网址:http://nodejs.cn/download/ 通过 ...

  2. 教你如何通过CodeArts IDE插件调用API,高效合成语音

    摘要:本实验基于华为云自研CodeArts IDE,指导用户通过使用华为云API,来实现一个文字合成语音的应用. 本文分享自华为云社区<通过CodeArts IDE插件调用API,高效合成语音! ...

  3. NodeJS学习三之API

    Node采用V8引擎处理JavaScript脚本,最大特点就是单线程运行,一次只能运行一个任务.这导致Node大量采用异步操作(asynchronous opertion),即任务不是马上执行,而是插 ...

  4. sublime text 安装nodejs开发插件

    系统:windows10nodejs版本:v6.1.14 请先配置好环境变量,这里就不说啦. 下载并设置nodejs插件 下载地址为https://github.com/tanepiper/Subli ...

  5. nodejs 各种插件

    __dirname:全局变量,存储的是文件所在的文件目录 __filename:全局变量,存储的是文件名 代码:dirname.jsconsole.log(__dirname); 运行node dir ...

  6. FullCalendar日历插件(中文API)

    FullCalendar提供了丰富的属性设置和方法调用,开发者可以根据FullCalendar提供的API快速完成一个日历日程的开发,本文将FullCalendar的常用属性和方法.回调函数等整理成中 ...

  7. PhpStorm插件之Api Debugger

    安装插件 File->Setting->Pluugins   搜索  Api Debugger 如何使用 安装完插件后,RESTART IDE,在编辑器右侧 即可找到最新安装的 Api D ...

  8. Elasticsearch--集群管理_别名&插件&更新API

    目录 使用索引别名 别名 创建别名 修改别名 合并命令 获取所有别名 移除别名 别名中过滤 别名和路由 Elasticsearch插件 基础知识 安装插件 移除插件 更新设置API 使用索引别名 通过 ...

  9. 【Azure 应用服务】NodeJS Express + MSAL 实现API应用Token认证(AAD OAuth2 idToken)的认证实验 -- passport.authenticate('oauth-bearer', {session: false})

    问题描述 在前两篇博文中,对NodeJS Express应用 使用MSAL + AAD实现用户登录并获取用户信息,获取Authorization信息 ( ID Token, Access Token) ...

  10. 笔记-Nodejs中的核心API之Events

    最近正在学习Node,在图书馆借了基本关于Node的书,同时在网上查阅资料,颇有收获,但是整体感觉对Node的理解还是停留在一个很模棱两可的状态.比如Node中的模块,平时练习就接触到那么几个,其他的 ...

随机推荐

  1. python学习总结(重要!!!)

    前取,后不取        index从0开始 list = [1,2,3,4,5,6,7,8,9]print(list[3:7]) #输出:[4, 5, 6, 7]print(list[3:-2]) ...

  2. jmeter测试udp广播(jmeter发送udp)

    jmeter测试udp广播(jmeter发送udp) jmeter测试udp广播(jmeter接收udp) 先下载安装第三方插件 下载链接:https://jmeter-plugins.org/ins ...

  3. Jmeter函数助手25-log

    log函数用于记录一条日志并返回其值. String to be logged (and returned):函数会返回该值.控制台也能看到该字符 Log level (default INFO) o ...

  4. 【SpringBoot】15 数据访问P3 整合Mybatis

    重新建立一个SpringBoot工程 选择依赖组件 然后删除不需要的Maven&Git文件 还是先查看我们的POM文件 整合Mybatis的组件多了这一个,默认的版本是3.5.4 然后再看看整 ...

  5. 【Vue】Re17 Router 第四部分(参数传递,守卫函数)

    一.案例搭建 新建Profile组件 组件写好内容后配置路由 { path : '/profile', component : () => import('../components/Profi ...

  6. 同策略强化学习算法可以使用经验缓存池(experience buffer)吗 ??? 设计一个基于缓存池的改进reinforce算法,给出初步的尝试 ---------- (reinforce + experience buffer)

    本文使用代码地址: https://gitee.com/devilmaycry812839668/reinforce_with_-experience-buffer ================= ...

  7. 解决GitHub不显示图片问题——GitHub图片显示失败

    在Github上打不开图片是一个极为常见的事情,出现这样的问题的原因就是我们的主机无法正确解析图片所在主机域名对应的IP地址,下面给出一个在本地主机(window10)上使用ping命令解析目标主机域 ...

  8. .NET 8 + Blazor 多租户、模块化、DDD框架、开箱即用

    前言 基于 .NET 8 的开源项目,主要使用 WebAPI + Blazor 支持多租户和模块化设计,DDD构建.可以帮助我们轻松地搭建起一个功能完善的Web应用程序.除了帮助你快速构建应用程序之外 ...

  9. 后缀数组--SA--字符串

    SA (Suffix Array) -- 后缀数组 简介 这里明白两个定义: \(SA_i\) : 按字典序排列后大小为 \(i\) 的后缀的后缀头的下标. \(Rank_i\) : 后缀头的下标为 ...

  10. Catlan--卡特兰数--组合数学

    卡特兰数 \(Catlan\) ·赘述 其实发现卡特兰数和之前不同的是,前面的是给你公式,让你去求具体的例子,然而卡特兰数这里是给你大量例子来给你证明和解释什么是卡特兰数. ·定义 对于卡特兰数来说, ...