最近因公司业务所需,研究起了Electron,其他一起都还顺利,但在调用C++这卡住了。
通过官方文档提供的方式编译生成了.node文件,官方文档地址:https://nodejs.org/api/addons.html(英文版可能需要翻墙,中文版内容不准)
在使用.node时,一直出现如下错误:
App threw an error during load
Error: The module '\\?\D:\work\src\electron\electron-quick-start\Addon\addon.node'
was compiled against a different Node.js version using
NODE_MODULE_VERSION 72. This version of Node.js requires
NODE_MODULE_VERSION 70. Please try re-compiling or re-installing
the module (for instance, using `npm rebuild` or `npm install`).
看字面意思是版本号不对,我仔细对了下我的nodejs版本号与Electron要求的版本号,没错啊都是12.0.0,问题在哪呢?
后来我仔细阅读Electron官方提供的资料,看到如下一段,链接:https://electronjs.org/docs/tutorial/using-native-node-modules#%E4%BD%BF%E7%94%A8-node-%E5%8E%9F%E7%94%9F%E6%A8%A1%E5%9D%97
顺着这个方向,又看到如下内容,链接:https://electronjs.org/docs/development/upgrading-node
再接着找到如下内容,链接:https://github.com/electron/node
这下终于明白了,原来Electron使用的Nodejs是自己内部的版本,所以只要使用nodejs官方发布版去编译的C++插件都会出现这个问题。
后来我把Electron分支Nodejs代码下下来看,发现有一段代码是去比对NODE_MODULE_VERSION这个版本号是否一致,如果不一致则会抛出以上错误
要解决这个问题,我想到了两个办法:
1.使用Electron分支下的nodejs版本代码去编译并使用,这个比较复杂,我没试过。
2.手动改写插件版本号,这个简单。
下面介绍下第二种方法:
我们自行编译.node插件时,引用到了AppData\Local\node-gyp\Cache\xxx\include\node目录下的node_version.h文件,文件中有NODE_MODULE_VERSION版本号的定义,将这个版本号改成和Electron使用的版本号一致,再编译使用方可通过。
如果不确定node_version.h文件位置的,可以用vs打开build生成的工程文件,利用vs找到NODE_MODULE宏定义,这时会跳到node.h头文件,接着找NODE_MODULE_X的定义,这时就会跳到node_version.h文件就能找到NODE_MODULE_VERSION的定义了。
注意,这么改有风险,因为Electron内嵌的Nodejs在官方发布版的基础上做过改动,所以尽量选择使用版本号差别不大并且v8的版本兼容的nodejs版本,或者对比官方代码,了解改动差别,主要接口没变的话基本就没啥问题。
因为总觉的以上方案不靠谱,在网上翻遍各种资料,终于找到完美解决方案,
node-gyp rebuild --target=5.0.5 --arch=ia32 --dist-url=https://npm.taobao.org/mirrors/atom-shell
--target指定electron版本号
--dist-url指定头文件下载地址,原答案这个值写的是https://atom.io/download/electron,国内的话不管用
加上这几个的意思从指定地址下载相应electron版本的头文件,然后用下载的头文件进行编译,下下来的头文件便是electron修改过的nodejs版本。
来自:
https://newsn.net/say/electron-gyp-param.html
https://www.cnblogs.com/mdorg/p/10417945.html