javascript - Node 串行端口模块无法与子进程 fork() 一起使用

标签 javascript node.js electron node-serialport

我正在尝试使用 Electron 中的 child_process 模块中的 fork() 在 USB 中进行异步读取。也就是说,当我单击 div (id="read") 时,一条消息会从渲染器进程发送到主进程,然后主进程作为响应调用 fork() 并异步从 USB 读取数据。为了处理 USB 通信,我使用 node-serialport library

index.html

<html>

    <head>
        <link rel="stylesheet" href="style.css">
        <link rel="stylesheet" href="scroll.css">
        <link href="https://fonts.googleapis.com/css?family=Questrial" rel="stylesheet">
    </head>

    <body>

        <div id="ui-container">


            <div id="top-bar" class="shadowed">
                <div id="top-bar-left-decoration"></div>
                <div id="close-app" class="top-bar-button"><div class="unselectable top-button-text">x</div></div> 
                <div id="maximize-app" class="top-bar-button"><div class="unselectable top-button-text">&#9633</div></div> 
                <div id="minimize-app" class="top-bar-button"><div class="unselectable top-button-text">-</div></div>
            </div>

            <div id="navigation-bar"> 
                <div id="deviceSelection" class ="unselectable navigation-button">

                    <img id="deviceSelection-image" class="navigation-image" src="Icons/selectDevice2.png">
                </div>

                <div id="info" class="unselectable navigation-button">

                    <img id="info-image" class="navigation-image" src="Icons/DeviceInfoNoBorder.png">
                </div>

                <div id="config" class=" unselectable navigation-button">

                    <img id="config-image" class="navigation-image" src="Icons/DeviceConfigNoBorder.png">
                </div>

                <div id="graph" class="unselectable navigation-button">

                    <img id="graph-image" class="navigation-image" src="Icons/DataViewNoBorder.png">
                </div>

                <div id="export" class="unselectable navigation-button">

                    <img id="export-image" class="navigation-image" src="Icons/Export.png">
                </div>

                <div id="options-container"><img id="options" src="Icons/options.png"></div> 

            </div>

            <div id="information-data-view" class="unselectable">
                <div id="data">
                        <div id="push-to-read"></div>
                </div>
            </div>

        </div>


    </body>

    <script src="ipcRenderer.js" type="text/javascript"></script>
</html>

这是我的index.js,负责创建浏览器窗口

const electron = require('electron')
const ipcMain = require('electron').ipcMain
const url = require('url')
const path = require('path')
const child_process = require('child_process')
const app = electron.app


var windowFullScreen = false;

const BrowserWindow = electron.BrowserWindow



var MainWindow;
app.on('ready', function()
{
    MainWindow = new BrowserWindow({
        width: 1024, 
        height: 768, 
        backgroundColor : '123355',
        frame: false,
        resizable: true,
        movable: true,
        show: false
    })

MainWindow.on('ready-to-show', () => {
    MainWindow.show()
    console.log('Ready to go!')
})
MainWindow.loadURL(url.format({
    pathname: path.join(__dirname, 'index.html'),
    protocol: 'file:',
    slashes: true
}))
ipcMain.on("minimize", (event, arg) => {
    MainWindow.minimize()
    console.log("minimize")
})
ipcMain.on("maximize", (event, arg) => {
    if (!windowFullScreen) {
        MainWindow.maximize()
        windowFullScreen = true;
    }
    else {
        MainWindow.setSize(800, 600)
        windowFullScreen = false;
    }

    console.log("maximize")
})
ipcMain.on("close", (event, arg) => {
    MainWindow.close()
    console.log("close")
})

ipcMain.on("read", (event, arg) => {
    setInterval(()=>{console.log(usbcomm.read(1))}, 1)
    const process = child_process.fork('./readUSB.js')
})
})

ipcRenderer.js

这是我为 DOM 设置 onclick 事件处理程序的地方。这里的相关部分是 read.onclick() ,它向主进程发送“读”消息,告诉它 fork 进程并运行我在下面包含的 readUSB.js 。

const electron = require('electron');
const ipcRenderer = electron.ipcRenderer;


var minimize = document.getElementById("minimize-app");
var maximize = document.getElementById("maximize-app");
var read = document.getElementById("push-to-read");
var close = document.getElementById("close-app");

minimize.onclick = (event) => {
    ipcRenderer.send("minimize", "an-argument")
    console.log("minimize");    
} 

maximize.onclick = (event) => {
    ipcRenderer.send("maximize", "an-argument")
    console.log("maximize")
} 

close.onclick = (event) => {
    ipcRenderer.send("close", "an-argument")
    console.log("close")
}

read.onclick = (event) => {
    ipcRenderer.send("read", "an-argument")
    console.log("read")
}

读取USB.js

这段代码应该简单地从USB读取一个字节并将其输出到控制台,它所做的只是不读取任何内容,并输出null(如果usbcomm.read()没有读取任何内容,则返回null)。这只是一个概念证明,以确保我可以异步读取(),但事情显然无法正常工作。

const SerialPort = require('serialport')
const usbcomm = new SerialPort("COM4")
usbcomm.setEncoding('utf-8')


usb_rx();


function usb_rx()
{

    console.log("running in child process!")
    console.log(usbcomm.read(1))

}

如果我将此代码作为事件处理程序运行以响应来自渲染器进程的“读取”消息,则一切都会正常运行,并且我会从 USB 读取正确的响应。我已经确认这不是我的 USB 设备的问题,并且在简单的用例中 usbcomm.read() 的行为符合预期。

我现在想知道 Node 串行端口是否出于某种原因无法与 fork() 内部运行的代码一起使用。

最佳答案

您可以从渲染器进行阻塞调用。只要将它们保持在 20-30 毫秒以下,并且不会因阻塞调用而淹没进程,就可以了。我们在渲染器进程中广泛使用了 node-serialportnode-ffi,每秒进行数十次调用,传输兆字节的数据,同时保持 UI 的响应速度,没有任何抖动。

也许您无法让阻塞调用保持足够短?如果在主进程中进行呢?这也是一个坏主意。阻止主进程会阻止渲染器和 GPU 进程之间的 IPC,因此您也可能会以这种方式挂起 UI。

如果您确实必须进行长时间的阻塞同步调用,请将它们放在隐藏的渲染器进程中。您可以根据需要阻止它,因为它不显示任何 UI。为了使这更容易,您可以使用 electron-remote - renderer-taskpool将长时间运行的任务转移到另一个渲染器进程上的功能。

关于javascript - Node 串行端口模块无法与子进程 fork() 一起使用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51934861/

相关文章:

windows - 在 windows 上与 node.js 或 karma 或其他任何东西一起使用时如何多次使用 cmd

node.js - expect(jest.fn()).toHaveBeenCalled() 失败,即使该函数已被调用

javascript - Node子进程打包Electron App后立即退出

javascript - jquery将数组对象转换为字符串

node.js - 在多核服务器中将 socket.io 与集群一起使用的好方法?

javascript - 如果 div 移动,Angularjs ng-repeat 不起作用

Electron - 从 View 调用函数

node.js - 如何正确找到 .txt 文件并使用 NodeJS 读取其内容

javascript - 使用 Jade Html 预处理器分别在每个 div 内包含一个 img 标签的多个 Div

javascript - 仅用一个滚动条在 X 轴上移动三个表格