为什么
如果我们的应用想要实现这样一个需求:监听电脑的usb接口,当有新的设备(移动硬盘或者U盘)接入电脑时,能够获取里面的移动设备的情况并更新到应用程序的界面上。
按照 Electron 或者 Node.js 现成的接口,我们无法直接实现。
这时候,我们就可以根据我们自己的情况,对系统的底层接口封装成独立的动态链接库(.dll),然后把动态链接库暴露给Electron 或者 Node.js 进行调用,从而实现需求。
简介
Node.js 可以通过对接 .dll,调用系统底层接口从而实现原有接口不提供的功能。同理,Electron 是基于Node.jss 进行封装的,Electron 也就拥有了对接动态链接库,能够把不可能变为可能。
方案选择
截至本文完成时间(2018-12-17)
现有的主流的方案一共有两项
-
自定义C++ Addons
-
node-ffi 对接方案
(暂不讨论 .net core 等对接方式)
原理分析
首先,明确一个观点,Electron 是基于 Node.js 封装的框架,Electron 对接 动态链接库 的底层方案,是基Node.js c++ Addons机制
从 Node.js 官网上可以了解到,Addons机制涉及多个组件和 API 的知识:
-
V8: Node.js 目前用于提供 JavaScript 实现的 C++ 库。 V8 提供了用于创建对象、调用函数等的机制。 V8 的 API 文档主要在
v8.h
头文件中(Node.js 源代码中的deps/v8/include/v8.h
),也可以在查看V8 在线文档。 -
libuv: 实现了 Node.js 的事件循环、工作线程、以及平台所有的的异步操作的 C 库。 它也是一个跨平台的抽象库,使所有主流操作系统中可以像 POSIX 一样访问常用的系统任务,比如与文件系统、socket、定时器、以及系统事件的交互。 libuv 还提供了一个类似 POSIX 多线程的线程抽象,可被用于强化更复杂的需要超越标准事件循环的异步插件。 建议插件开发者多思考如何通过在 libuv 的非阻塞系统操作、工作线程、或自定义的 libuv 线程中降低工作负载来避免在 I/O 或其他时间密集型任务中阻塞事件循环。
-
内置的 Node.js 库:Node.js 自身开放了一些插件可以使用的 C++ API。 其中最重要的是
node::ObjectWrap
类。 -
Node.js 包含一些其他的静态链接库:如 OpenSSL,这些库位于 Node.js 源代码中的
deps/
目录。 只有 V8 和 OpenSSL 符号是被 Node.js 开放的,