Springboot + rxtx 实现串口读写 案例

article/2025/10/31 16:05:10

经过大量翻阅文章总结出springboot连接串口最可行的方法!希望能帮到大家( •̀ ω •́ )✧

使用Rxtx实现串口通信

1、配置pom.xml

		<!-- 串口内容读取 --><dependency><groupId>org.bidib.jbidib.org.qbang.rxtx</groupId><artifactId>rxtxcomm</artifactId><version>2.2</version></dependency><!-- 测试 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-test</artifactId><scope>test</scope></dependency><dependency><groupId>junit</groupId><artifactId>junit</artifactId><scope>test</scope></dependency>

2、创建串口类

@Slf4j
@Data
public class SerialPortEntity {private String portName;//端口号private Integer baudrate;//波特率private int databits;//数据位private String databitsStr;//数据位字符串private int stopbits;//停止位private String stopbitsStr;//停止位字符串private int paritybit;//校验位private String paritybitStr;//校验位字符串private SerialPort serialPort = null; //串口会话public SerialPortEntity(String portName, Integer baudrate, String paritybit, String databits, String stopbits){this.portName = portName;this.baudrate = baudrate;this.databitsStr = databits;this.stopbitsStr = stopbits;this.paritybitStr = paritybit;//校验位switch (paritybit) {case "NONE":this.paritybit = SerialPort.PARITY_NONE;break;case "ODD":this.paritybit = SerialPort.PARITY_ODD;break;case "EVEN":this.paritybit = SerialPort.PARITY_EVEN;break;case "MARK":this.paritybit = SerialPort.PARITY_MARK;break;case "SPACE":this.paritybit = SerialPort.PARITY_SPACE;break;}//数据位switch (databits) {case "5":this.databits = SerialPort.DATABITS_5;break;case "6":this.databits = SerialPort.DATABITS_6;break;case "7":this.databits = SerialPort.DATABITS_7;break;case "8":this.databits = SerialPort.DATABITS_8;break;}//停止位switch (stopbits) {case "1":this.stopbits = SerialPort.STOPBITS_1;break;case "1.5":this.stopbits = SerialPort.STOPBITS_1_5;break;case "2":this.stopbits = SerialPort.STOPBITS_2;break;}}// 连接串口public SerialPort connect(){// 查看所有串口SerialPortUtil serialPortUtil = SerialPortUtil.getSerialPortUtil();List<String> portList = serialPortUtil.findPort();log.info("发现全部端口: "+portList);log.info("尝试打开端口:"+portName+" ....");// 打开指定端口SerialPort serialPort = serialPortUtil.openPort(portName,baudrate,databits,paritybit,stopbits);SerialPortListener listener = new SerialPortListener();listener.setSerialPort(serialPort);setSerialPort(serialPort);serialPortUtil.addListener(serialPort,listener);return serialPort;}// 关闭串口public void close(){SerialPortUtil serialPortUtil = SerialPortUtil.getSerialPortUtil();if(serialPort != null) serialPortUtil.closePort(serialPort);else log.info("请先调用connect方法!");}// 发送指令public void send(byte[] code){SerialPortUtil serialPortUtil = SerialPortUtil.getSerialPortUtil();if(serialPort != null) serialPortUtil.sendToPort(serialPort,code);else log.info("请先调用connect方法!");}}

3、创建监听器

@Slf4j
@Data
public class SerialPortListener implements SerialPortEventListener {private SerialPort serialPort = null;@Overridepublic void serialEvent(SerialPortEvent serialPortEvent) {switch (serialPortEvent.getEventType()){// 串口存在有效数据case SerialPortEvent.DATA_AVAILABLE:byte[] bytes = SerialPortUtil.getSerialPortUtil().readFromPort(serialPort);log.info("===========start===========");log.info(new Date() + "【读到的字符】:-----" + Arrays.toString(bytes));log.info(new Date() + "【字节数组转16进制字符串】:-----" + ModBusUtils.bytes2HexString(bytes));log.info("===========end===========");break;// 2.输出缓冲区已清空case SerialPortEvent.OUTPUT_BUFFER_EMPTY:log.error("输出缓冲区已清空");break;// 3.清除待发送数据case SerialPortEvent.CTS:log.error("清除待发送数据");break;// 4.待发送数据准备好了case SerialPortEvent.DSR:log.error("待发送数据准备好了");break;// 10.通讯中断case SerialPortEvent.BI:log.error("与串口设备通讯中断");break;default:break;}}
}

4、创建串口工具类

@Slf4j
public class SerialPortUtil {private static SerialPortUtil serialPortUtil = null;static {serialPortUtil = new SerialPortUtil();}private SerialPortUtil(){}/*** 获取提供服务的SerialTool对象* @return serialPortUtil*/public static SerialPortUtil getSerialPortUtil(){if(serialPortUtil == null){serialPortUtil = new SerialPortUtil();}return serialPortUtil;}/*** 查找所有可用端口* @return 可用端口名称列表*/public ArrayList<String> findPort() {// 获得当前所有可用串口Enumeration<CommPortIdentifier> portList = CommPortIdentifier.getPortIdentifiers();ArrayList<String> portNameList = new ArrayList<>();// 将可用串口名添加到List并返回该Listwhile (portList.hasMoreElements()) {String portName = portList.nextElement().getName();portNameList.add(portName);}return portNameList;}/*** 打开串口* @param portName 端口名称* @param baudrate 波特率  9600* @param databits 数据位  8* @param parity   校验位(奇偶位)  NONE :0* @param stopbits 停止位 1* @return 串口对象*/public SerialPort openPort(String portName, int baudrate, int databits, int parity, int stopbits) {try {// 通过端口名识别端口CommPortIdentifier portIdentifier = CommPortIdentifier.getPortIdentifier(portName);// 打开端口,并给端口名字和一个timeout(打开操作的超时时间)CommPort commPort = portIdentifier.open(portName, 5000);// 判断是不是串口if (commPort instanceof SerialPort) {SerialPort serialPort = (SerialPort) commPort;// 设置一下串口的波特率等参数serialPort.setSerialPortParams(baudrate, databits, stopbits, parity);log.info("打开串口 " + portName + " 成功 !");return serialPort;} else {log.error("不是串口");}} catch (NoSuchPortException e1) {log.error("没有找到端口");e1.printStackTrace();} catch (PortInUseException e2) {log.error("端口被占用");e2.printStackTrace();} catch (UnsupportedCommOperationException e) {e.printStackTrace();}return null;}/*** 关闭串口* @param serialPort 待关闭的串口对象*/public void closePort(SerialPort serialPort) {if (serialPort != null) {serialPort.close();}}/*** 往串口发送数据* @param serialPort 串口对象*/public void sendToPort(SerialPort serialPort, byte[] bytes) {OutputStream out = null;try {out = serialPort.getOutputStream();out.write(bytes);out.flush();} catch (IOException e) {e.printStackTrace();} finally {try {if (out != null) {out.close();}} catch (IOException e) {e.printStackTrace();}}}/*** 从串口读取数据* @param serialPort 当前已建立连接的SerialPort对象* @return 读取到的数据*/public byte[] readFromPort(SerialPort serialPort) {InputStream in = null;byte[] bytes = null;try {Thread.sleep(500);} catch (InterruptedException e) {e.printStackTrace();}try {in = serialPort.getInputStream();// 获取buffer里的数据长度int bufferlength = in.available();while (bufferlength != 0) {// 初始化byte数组为buffer中数据的长度bytes = new byte[bufferlength];in.read(bytes);bufferlength = in.available();}} catch (IOException e) {e.printStackTrace();} finally {try {if (in != null) {in.close();}} catch (IOException e) {e.printStackTrace();}}return bytes;}/*** 添加监听器* @param port     串口对象* @param listener 串口监听器*/public void addListener(SerialPort port, SerialPortEventListener listener) {try {// 给串口添加监听器port.addEventListener(listener);// 设置当有数据到达时唤醒监听接收线程port.notifyOnDataAvailable(true);// 设置当通信中断时唤醒中断线程port.notifyOnBreakInterrupt(true);} catch (TooManyListenersException e) {log.error("太多监听器");e.printStackTrace();}}/*** 删除监听器** @param port     串口对象* @param listener 串口监听器*/public void removeListener(SerialPort port, SerialPortEventListener listener) {// 删除串口监听器port.removeEventListener();}/*** 设置串口的Listener** @param serialPort* @param listener*/public static void setListenerToSerialPort(SerialPort serialPort, SerialPortEventListener listener) {try {// 给串口添加事件监听serialPort.addEventListener(listener);} catch (TooManyListenersException e) {e.printStackTrace();}// 串口有数据监听serialPort.notifyOnDataAvailable(true);// 中断事件监听serialPort.notifyOnBreakInterrupt(true);}
}

5、测试类

@SpringBootTest
@Slf4j
public class RxtxTest {@Testpublic void SerialDemo(){SerialPortEntity serialPortEntity = new SerialPortEntity("COM4",19200,"NONE","8","1");try {SerialPort serialPort = serialPortEntity.connect();serialPortEntity.send(new byte[]{1,3,0,38,0,1,101,(byte)193});Thread.sleep(10000);serialPortEntity.close();}catch (Exception e){serialPortEntity.close();e.printStackTrace();}}
}

好啦!现在来测试一下( •̀ ω •́ )✧~

准备调试工具

串口调试工具:http://www.zlmcu.com/document/com_debug_tools.html在这里插入图片描述

虚拟串口工具 :https://download.csdn.net/download/qq_37281772/87449453

在这里插入图片描述

操作步骤
1、打开虚拟串口工具,创建虚拟串口对,我使用的是COM4和COM5(端口可自定义)
2、打开串口调试工具,打开COM5串口(因为上面测试代码打开了COM4)
3、启动Test代码

COM4给COM5下发指令

在这里插入图片描述在这里插入图片描述

COM5给COM4下发指令

在这里插入图片描述在这里插入图片描述

萌新第一次写文,如文章有误,欢迎指出哈ヾ(>▽<)o*

参考文献:
1、Netty介绍:https://www.cnblogs.com/MrRightZhao/p/11925307.html
2、Netty集成串口RXTX编程:https://www.jianshu.com/p/9e9747dedc3d
3、Java使用Netty实现Modbus-RTU通信协议:https://blog.csdn.net/qq_40042416/article/details/128457288
4、springboot+netty+串口:https://blog.csdn.net/csweldn520/article/details/116661871
5、java利用socket通信实现Modbus-RTU通信协议:https://blog.csdn.net/qq_40042416/article/details/115672989


http://chatgpt.dhexx.cn/article/VMIekOcU.shtml

相关文章

java使用RXTX的详细总结

1 简介 项目要求&#xff1a;读取串口信息&#xff0c;并输出展示相应的图像&#xff0c;本篇文章不深入谈前端的内容着重于后端遇到的问题。2021-01-08更新&#xff0c;发现一个重大的问题&#xff1a;如果有人能解决的话请在评论区回复&#xff1a; 在连上串口后&#xff0c…

JAVA使用RXTXcomm进行串口通信(一)

1.引入RXTXcomm.JAR包 首先下载相应的jar文件 压缩包包括:RXTXcomm.jar(64位环境)、win32com.dll和javax.comm.properties。 下载地址:https://www.aliyundrive.com/s/JSeSQsAyYeZ 点击链接保存&#xff0c;或者复制本段内容&#xff0c;打开「阿里云盘」APP &#xff0c;无需…

SpringBoot使用RXTX连接串口教程及遇到的坑总结

文章目录 SpringBoot使用RXTX连接串口教程及遇到的坑总结一、所用环境及依赖二、部署流程2.1 下载RXTXComm包2.2 部署RXTXComm包 三、编写串口使用程序3.1 编写RXTXConfig.java3.2 编写实体类SerialPortEntity3.3 编写监听器SerialPortListener3.4 编写工具类SerialPortUtil3.5…

【计算机基础|计算机组成原理】【10】海明校验码

海明校验码 海明校验码思路 偶校验&#xff1a;1010 → 01010&#xff0c;能发现奇数位错误&#xff0c;但无法确定是哪一位出错 → 1个校验位只能携带2种状态信息&#xff1a;对/错 海明码设计思路&#xff1a;将信息位分组进行偶校验 → 多个校验位能携带多种状态信息&#x…

计算机组成原理汉明校验,海明校验码(计算机组成原理11)

海明校验码 视频链接地址&#xff1a; https://www.bilibili.com/video/BV1BE411D7ii?fromsearch&seid6420326887479343502 前言 在本篇中&#xff0c;你将掌握 海明码的基本思想 海明码的求解步骤和全校验码 在计算机内部进行存储、计算的数据都是以二进制形式传送&#…

海明校验码纠错设计原理

本文不阐述海明码计算方式&#xff0c;为了节省您的时间&#xff0c;请先学习海明码/奇偶校验是如何计算的&#xff0c;如果好奇其海明码的纠错设计思路再来看此文章&#xff01; 信息为位数为n&#xff0c;校验位数为k&#xff0c;正确状态占1位&#xff0c;所以总位数&#…

2.21 海明校验码

海明校验码 需要了解海明码的编码规则&#xff0c;要会计算需要多少位校验位。 海明码的编码规则 校验位&#xff1a; 校验位的位置是有规律的。都是位于2n。 比如20(1),21(2),22(4)。。都是校验位。 信息位&#xff1a;不是校验位的其他位置。 举个例子 当信息位有1位&am…

【软考学习7】数据校验——海明校验码、循环校验码、奇偶校验码

一、检错、纠错和码距 1.1 检错 从接收的报文中&#xff0c;检查出错误。 1.2 纠错 从接收的报文中检查出错误&#xff0c;并改正错误。 一般通过加冗余信息&#xff08;增大码距&#xff09;来实现。 1.3 码距 码距是整个编码系统中任意两个码字的最小距离。 也就是说&a…

超详细的海明校验码方法解读

海明校验码原理&#xff1a;在有效的信息为中加入几个校验位形成海明码&#xff0c;使码距[rjazgj1] 比较均匀地拉大&#xff0c;并把海明码的每个二进制位分配到几个奇偶校验组[rjazgj2] 中。当某一位出错后&#xff0c;就会引起有关的几个校验位的值发生变化&#xff0c;这不…

海明校验码原理以及作用机制的介绍

什么是海明校验码&#xff1f; 由Richard Hamming于1950年提出、还被广泛采用的一种很有效的校验方法&#xff0c;是只要增加少数几个校验位&#xff0c;就能检测出二位同时出错、亦能检测出一位出错并能自动恢复该出错位的正确值的有效手段&#xff0c;后者被称为自动纠错。 它…

海明校验码的计算及检验

海明校验码的计算及检验 目录 海明校验码的计算及检验知识背景计算海明校验码步骤一&#xff1a;计算校验码位数步骤二&#xff1a;确定校验组步骤三&#xff1a;计算校验码的值得出海明校验码 利用海明校验码校验数据其他 总结 最近和兄弟探讨一个海明校验码的题目&#xff0c…

海明校验码举例

海明校验码举例&#xff1a; 编制ASCII字符M的海明校验码。 解&#xff1a;M的ASCII码为A6A5A4A3A2A1A01001101 M为7位那么海明码最少需要2i&#xff0c;也就是说需要&#xff0c;才能表示出来&#xff0c;&#xff08;238&#xff09; 用哪些信息位分别被哪些校验位效验如…

计算机底层:海明校验码。

计算机底层&#xff1a;海明校验码。 海明校验码是由奇偶校验码中的偶校验延申出来的&#xff1a; 计算机底层&#xff1a;奇偶校验码_srhqwe的博客-CSDN博客 了解海明校验码之前需要先了解奇偶校验码。 海明校验码设计思路&#xff1a; 需要知道&#xff1a;多个校验位就能携…

海明校验码

1. 海明码的特点&#xff1a; 其中m表示数据位的位数&#xff0c;k表示海明校验码的位数 k位海明校验码一共可以表示种校验信息结果&#xff0c;其中有一种要用来表示没有出错的情况&#xff0c;则其余还剩-1种结果&#xff0c;为了使校验结果可以指出任一位出错的位置&#x…

计算机组成原理学习笔记:海明校验码

概述 海明校验码又可以称为汉明校验码, 这只是一个音译的问题, 作者是 Richard Hamming海明校验码对于信息纠错这个领域的贡献十分巨大&#xff0c;Richard Hamming 获得了1968年的图灵奖内容主要包括&#xff1a;海明校验码的思想、如何构建海明校验码、如何使用海明校验码 …

海明码校验【简单详细】

海明码 1.什么是海明码: 一个名叫Richard Hanming老爷爷在1950年提出的检验纠错方法&#xff0c;它具有一位纠错能力。 2.海明码的计算方法: 设欲检测的二进制代码为n位,K为检测位(提供纠错),总共nk位代码 当中检测位满足的关系: 2 k 2^{k} 2k>(nk1) 此关系也是求不同代码长…

一文看懂海明校验码及其计算方法(详细总结)

网上看了好几篇文章后终于算是捋明白了&#xff0c;但是看到的这些资源要么说得云里雾里&#xff0c;要么干脆说得有问题&#xff08;然后还被点了好多赞。。。&#xff09;&#xff0c;无论如何这些都容易误导小白。作为C站多年老潜水员&#xff0c;我还是把海明校验码的要点总…

ResNets

ResNets 背景&#xff1a; 非常非常深的神经网络是很难训练的&#xff0c;因为存在梯度消失和梯度爆炸问题。 《转载更改》 https://blog.csdn.net/qq_29893385/article/details/81207203 ResNets是由残差块&#xff08;Residual block&#xff09;构建的 首先解释一下什么是…

正确定位混淆后Crash代码行数

Android--定位混淆后Crash代码行数 一、需求背景二、前期准备三、对混淆日志进行还原四、示例 一、需求背景 打包时需要对代码进行混淆&#xff0c;目的是增加安全性&#xff0c;防⽌反编译。但这会导致App崩溃时&#xff0c;抓到的日志堆栈中显示的代码行数对应不上&#xff…

repalce

1、replace基本用法 <script>/*要求将字符串中所有的a全部用A代替*/var str "javascript is great script language!";//只会将第一个匹配到的a替换成Aconsole.log(str.replace("a", "A")); // > jAvascript is great script language…