基于纯JavaScript实现的MODBUS-RTU(串口和TCP) modbus-serial

2024-04-29 09:36

本文主要是介绍基于纯JavaScript实现的MODBUS-RTU(串口和TCP) modbus-serial,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

modbus-serial

如果你需要使用JavaScript来操作一台RS458的设备,那么你一定不能错过这个库 modbus-serial

安装和使用

npm install modbus-serial

支持的功能码

功能码函数
FC1 读取读线圈寄存器readCoils(coil, len)
FC2 读离散输入寄存器readDiscreteInputs(addr, arg)
FC3 读保持寄存器readHoldingRegisters(addr, len)
FC4 读输入寄存器readInputRegisters(addr, len)
FC5 写单个线圈寄存器writeCoil(coil, binary)
FC6 写单个保持寄存器writeRegister(addr, value)
FC15 写多个线圈寄存器writeCoils(addr, valueAry)
FC16 写多个保持寄存器writeRegisters(addr, valueAry)
FC43/14 读保持寄存器readDeviceIdentification(id, obj)

客户端串行:

  • modbus-RTU (SerialPort):通过串行线路,需要node serialport。

  • modbus-RTU (RTUBufferedPort):通过缓冲串行线路,需要node serialport。

  • modbus-ASCII (AsciiPort):通过串行线路,需要node serialport。

客户端TCP:

  • modbus-TCP (TcpPort):通过TCP/IP线路。

  • modbus-RTU (UdpPort):通过C701服务器,商业UDP到串行桥。

  • modbus-RTU (TcpRTUBufferedPort):通过TCP/IP线路,TCP/IP串行RTU缓冲设备。

  • modbus-RTU (TelnetPort):通过Telnet服务器,TCP/IP串行桥。

服务器

  • modbus-TCP (ServerTCP):通过TCP/IP线路。

示例

读取和写入

// create an empty modbus client
const ModbusRTU = require("modbus-serial");
const client = new ModbusRTU();// open connection to a serial port
client.connectRTUBuffered("/dev/ttyUSB0", { baudRate: 9600 }, write);function write() {client.setID(1);// write the values 0, 0xffff to registers starting at address 5// on device number 1.client.writeRegisters(5, [0 , 0xffff]).then(read);
}function read() {// read the 2 registers starting at address 5// on device number 1.client.readHoldingRegisters(5, 2).then(console.log);
}

读取多个从站

const ModbusRTU = require("modbus-serial");
// create an empty modbus client
const client = new ModbusRTU();
// open connection to a serial port
client.connectRTUBuffered("/dev/ttyS0", { baudRate: 9600 });
// set timeout, if slave did not reply back
client.setTimeout(500);// list of meter's id
const metersIdList = [10, 11, 12, 13, 14];const getMetersValue = async (meters) => {try{// get value of all metersfor(let meter of meters) {// output value to consoleconsole.log(await getMeterValue(meter));// wait 100ms before get another deviceawait sleep(100);}} catch(e){// if error, handle them here (it should not)console.log(e)} finally {// after get all data from slave, repeat it againsetImmediate(() => {getMetersValue(metersIdList);})}
}const getMeterValue = async (id) => {try {// set ID of slaveawait client.setID(id);// read the 1 registers starting at address 0 (first register)let val =  await client.readInputRegisters(0, 1);// return the valuereturn val.data[0];} catch(e){// if error return -1return -1}
}const sleep = (ms) => new Promise(resolve => setTimeout(resolve, ms));// start get value
getMetersValue(metersIdList);

ModbusTCP 服务

// create an empty modbus client
const ModbusRTU = require("modbus-serial");
const vector = {getInputRegister: function(addr, unitID) {// Synchronous handlingreturn addr;},getHoldingRegister: function(addr, unitID, callback) {// Asynchronous handling (with callback)setTimeout(function() {// callback = function(err, value)callback(null, addr + 8000);}, 10);},getCoil: function(addr, unitID) {// Asynchronous handling (with Promises, async/await supported)return new Promise(function(resolve) {setTimeout(function() {resolve((addr % 2) === 0);}, 10);});},setRegister: function(addr, value, unitID) {// Asynchronous handling supported also hereconsole.log("set register", addr, value, unitID);return;},setCoil: function(addr, value, unitID) {// Asynchronous handling supported also hereconsole.log("set coil", addr, value, unitID);return;},readDeviceIdentification: function(addr) {return {0x00: "MyVendorName",0x01: "MyProductCode",0x02: "MyMajorMinorRevision",0x05: "MyModelName",0x97: "MyExtendedObject1",0xAB: "MyExtendedObject2"};}
};// set the server to answer for modbus requests
console.log("ModbusTCP listening on modbus://0.0.0.0:8502");
const serverTCP = new ModbusRTU.ServerTCP(vector, { host: "0.0.0.0", port: 8502, debug: true, unitID: 1 });serverTCP.on("socketError", function(err){// Handle socket error if needed, can be ignoredconsole.error(err);
});

市面上有非常多的设备时 RS485,如果你之前是前端,并且想接触硬件,就学习一下它吧。

这篇关于基于纯JavaScript实现的MODBUS-RTU(串口和TCP) modbus-serial的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/945733

相关文章

Spring Boot2 系列教程(二十五)Spring Boot 整合 Jpa 多数据源

本文是 Spring Boot 整合数据持久化方案的最后一篇,主要和大伙来聊聊 Spring Boot 整合 Jpa 多数据源问题。在 Spring Boot 整合JbdcTemplate 多数据源、Spring Boot 整合 MyBatis 多数据源以及 Spring Boot 整合 Jpa 多数据源这三个知识点中,整合 Jpa 多数据源算是最复杂的一种,也是很多人在配置时最容易出错的一种。本

Flink中异步AsyncIO的实现 (源码分析)

先上张图整体了解Flink中的异步io 阿里贡献给flink的,优点就不说了嘛,官网上都有,就是写库不会柱塞性能更好 然后来看一下, Flink 中异步io主要分为两种   一种是有序Ordered   一种是无序UNordered 主要区别是往下游output的顺序(注意这里顺序不是写库的顺序既然都异步了写库的顺序自然是无法保证的),有序的会按接收的顺序继续往下游output发送,无序

采用java+B/S开发的全套医院绩效考核系统源码springboot+mybaits 医院绩效考核系统优势

采用java开发的全套医院绩效考核系统源码springboot+mybaits  医院绩效考核系统优势 医院绩效管理系统解决方案紧扣新医改形势下医院绩效管理的要求,以“工作量为基础的考核方案”为核心思想,结合患者满意度、服务质量、技术难度、工作效率、医德医风等管理发展目标的考核体系,形成医院的内部绩效考核与分配机制,通过信息化手段为绩效考评管理人员实施医院绩效考评工作提供了有效工具,扩展了信

【R语言篇】潜在类别分析(Latent Class Analysis, LCA)及其在R语言中的实现

1. 潜在类别分析简介 潜在类别分析(LCA)是一种用于识别样本中潜在子群的统计方法。这种方法基于观察到的数据来估计个体属于不同潜在类别的概率。LCA广泛应用于社会科学、市场研究、生物医学等领域,尤其适用于处理分类数据。 2. LCA的核心思想 LCA的核心在于假设数据中存在若干未观测到的、互斥的类别,每个类别代表了一种特定的群体特征。通过分析,我们可以估计出每个个体属于各个潜在类别的概率,进而

2024OD机试卷-绘图机器 (java\python\c++)

题目:绘图机器 题目描述 绘图机器的绘图笔初始位置在原点(0,0)机器启动后按照以下规则来进行绘制直线。 尝试沿着横线坐标正向绘制直线直到给定的终点E期间可以通过指令在纵 坐标轴 方向进行偏移,offsetY为正数表示正向偏移,为负数表示负向偏移 给定的横坐标终点值E 以及若干条绘制指令, 请计算绘制的直线和横坐标轴以及x=E的直线组成的图形面积。 输入描述 首行为两个整数 N 和 E

【Java并发基础】管程简介

前言 在Java 1.5之前,Java语言提供的唯一并发语言就是管程,Java 1.5之后提供的SDK并发包也是以管程为基础的。除了Java之外,C/C++、C#等高级语言也都是支持管程的。 那么什么是管程呢? 见名知意,是指管理共享变量以及对共享变量操作的过程,让它们支持并发。翻译成Java领域的语言,就是管理类的状态变量,让这个类是线程安全的。 synchronized关键字和wait(

基于Java+HttpClient+TestNG的接口自动化测试框架(四)-------参数存取处理

在真正开始接口测试之前,我们需要对参数的处理进行梳理。这里所说的“参数”,既包含之前在xml中的配置(我们称之为全局参数),也包含在每一条用例中书写的param。全局参数为固定不变的,而根据接口相应获取的数据是动态变化的。 之前我们已经用${param_name}的形式,定义了如何从公共参数池中调用所需要的参数。 并且当一个接口的请求发送完成,进行下一个接口操作的时候,我们也需要对相应

实战_Spring_Cloud

目录 前言 开发环境源码地址创建工程服务注册中心(Eureka) Eureka ServerEureka Client注册中心高可用小结负载均衡(Ribbon) RestTemplate调用负载均衡调用应用名称调用小结声明式服务调用(Feign) 服务端实现客户端实现小结统一配置中心(Config) Config Server向服务中心注册服务提供端改造配置动态刷新配置 Webhook小结异步

Spring Boot 2.X(十九):集成 mybatis-plus 高效开发

前言 之前介绍了 SpringBoot 整合 Mybatis 实现数据库的增删改查操作,分别给出了 xml 和注解两种实现 mapper 接口的方式;虽然注解方式干掉了 xml 文件,但是使用起来并不优雅,本文将介绍 mybats-plus 的常用实例,简化常规的 CRUD 操作。 mybatis-plus MyBatis-Plus(简称 MP)是一个 MyBatis 的增强工具,在 MyB

【吊打面试官系列】Java高并发篇 - 如何创建守护线程?

大家好,我是锋哥。今天分享关于 【如何创建守护线程?】面试题,希望对大家有帮助; 如何创建守护线程? 使用 Thread 类的 setDaemon(true)方法可以将线程设置为守护线程,需要注意的是,需要在调用 start()方法前调用这个方法,否则会抛出 IllegalThreadStateException 异常。