最近笔者接触到串口编程,网上搜了些资料,顺便整理一下。网上都在推荐使用Java RXTX开源类库,它提供了Windows、Linux等不同操作系统下的串口和并口通信实现,遵循GNU LGPL协议。看起来不错,写个例子试试。

准备运行环境

下载RXTX

RXTX下载地址是:http://fizzed.com/oss/rxtx-for-java

笔者操作系统是Windows10,下载对应版本的压缩包,解压后复制RXTXcomm.jar到D:\Program Files\Java\jdk1.8.0_152\jre\lib\ext目录下;复制rxtxParallel.dll和rxtxSerial.dll到D:\Program Files\Java\jdk1.8.0_152\jre\bin目录下。

注意:安装jdk时可能也顺便装了jre,需要复制到jdk的jre目录下。

下载Virtual Serial Port Driver

Virtual Serial Port Driver是一款非常好用的虚拟串口模拟软件,可以在计算机模拟串口,方便开发和测试。安装后打开界面如下:

可以看到右侧默认出现COM1和COM2的串口,点击Add pair就可以创建这两个串口了,打开计算机管理,可以看到本机多了这两个端口,如下图所示:

创建项目

创建serialPort项目,如下图所示:

源代码地址:https://github.com/wu-boy/serialPort.git

文中所用软件工具等资料下载:https://download.csdn.net/download/wu_boy/14003992

串口工具类

现在可以写一个串口工具类,方便开发和测试,代码如下:

public class SerialPortUtils {

    private static Logger log = LoggerFactory.getLogger(SerialPortUtils.class);

    /**
* 打卡串口
* @param portName 串口名
* @param baudRate 波特率
* @param dataBits 数据位
* @param stopBits 停止位
* @param parity 校验位
* @return 串口对象
*/
public static SerialPort open(String portName, Integer baudRate, Integer dataBits,
Integer stopBits, Integer parity) {
SerialPort result = null;
try {
// 通过端口名识别端口
CommPortIdentifier identifier = CommPortIdentifier.getPortIdentifier(portName);
// 打开端口,并给端口名字和一个timeout(打开操作的超时时间)
CommPort commPort = identifier.open(portName, 2000);
// 判断是不是串口
if (commPort instanceof SerialPort) {
result = (SerialPort) commPort;
// 设置一下串口的波特率等参数
result.setSerialPortParams(baudRate, dataBits, stopBits, parity);
log.info("打开串口{}成功", portName);
}else{
log.info("{}不是串口", portName);
}
} catch (Exception e) {
log.error("打开串口{}错误", portName, e);
}
return result;
} /**
* 串口增加数据可用监听器
* @param serialPort
* @param listener
*/
public static void addListener(SerialPort serialPort, DataAvailableListener listener) {
if(serialPort == null){
return;
}
try {
// 给串口添加监听器
serialPort.addEventListener(new SerialPortListener(listener));
// 设置当有数据到达时唤醒监听接收线程
serialPort.notifyOnDataAvailable(Boolean.TRUE);
// 设置当通信中断时唤醒中断线程
serialPort.notifyOnBreakInterrupt(Boolean.TRUE);
} catch (TooManyListenersException e) {
log.error("串口{}增加数据可用监听器错误", serialPort.getName(), e);
}
} /**
* 从串口读取数据
* @param serialPort
* @return
*/
public static byte[] read(SerialPort serialPort) {
byte[] result = {};
if(serialPort == null){
return result;
}
InputStream inputStream = null;
try {
inputStream = serialPort.getInputStream(); // 缓冲区大小为1个字节,可根据实际需求修改
byte[] readBuffer = new byte[7]; int bytesNum = inputStream.read(readBuffer);
while (bytesNum > 0) {
result = ArrayUtil.addAll(result, readBuffer);
bytesNum = inputStream.read(readBuffer);
}
} catch (IOException e) {
log.error("串口{}读取数据错误", serialPort.getName(), e);
} finally {
IoUtil.close(inputStream);
}
return result;
} /**
* 往串口发送数据
* @param serialPort
* @param data
*/
public static void write(SerialPort serialPort, byte[] data) {
if(serialPort == null){
return;
}
OutputStream outputStream = null;
try {
outputStream = serialPort.getOutputStream();
outputStream.write(data);
outputStream.flush();
} catch (Exception e) {
log.error("串口{}发送数据错误", serialPort.getName(), e);
} finally {
IoUtil.close(outputStream);
}
} /**
* 关闭串口
* @param serialPort
*/
public static void close(SerialPort serialPort) {
if (serialPort != null) {
serialPort.close();
log.warn("串口{}关闭", serialPort.getName());
}
} /**
* 查询可用端口
* @return 串口名List
*/
public static List<String> listPortName() {
List<String> result = new ArrayList<>(); // 获得当前所有可用端口
Enumeration<CommPortIdentifier> serialPorts = CommPortIdentifier.getPortIdentifiers();
if(serialPorts == null){
return result;
} // 将可用端口名添加到List并返回该List
while (serialPorts.hasMoreElements()) {
result.add(serialPorts.nextElement().getName());
}
return result;
}
}

测试代码

测试代码如下,先不要着急运行,下一步打开串口调试助手协助测试。

public class SerialPortTest {

    public static void main(String[] args) throws Exception{

        // 打开串口
SerialPort serialPort = SerialPortUtils.open("COM1", 9600, SerialPort.DATABITS_8,
SerialPort.STOPBITS_1, SerialPort.PARITY_NONE); // 监听串口读取数据
SerialPortUtils.addListener(serialPort, () -> {
byte[] data = SerialPortUtils.read(serialPort);
System.out.println(HexUtil.encodeHexStr(data));
}); // 往串口发送数据
byte[] data = {1, 2, 3};
SerialPortUtils.write(serialPort, data); /*// 关闭串口
Thread.sleep(2000);
SerialPortUtils.close(serialPort);*/ // 测试可用端口
//SerialPortUtils.listPortName().forEach(o -> System.out.println(o));
}
}

串口调试助手

UartAssist是一款很好用的串口调试助手,先运行串口调试助手,接收设置和发送设置都选择HEX,串口号选择COM2->COM1(测试代码使用的COM1),其他默认,点击打开串口,然后运行测试代码SerialPortTest,效果如下图所示:

运行测试代码后,串口调试助手显示收到01 02 03,然后串口调试助手点击发送,idea控制台也会显示收到11223344556677,说明COM1和COM2串口互相发送和接收数据成功。

粘包/拆包的解决方案

在实际应用中,有些功能复杂的串口通信可能会发生粘包/拆包的情况,这时可以自建一个缓冲区,用来缓冲数据并处理数据。《Netty权威指南第2版》中,有TCP粘包/拆包问题的解决之道,原理可供参考,需要自己写代码实现,推荐使用Netty的缓冲区ByteBuf,功能强大。

参考资料

1、使用Java实现串口通信(二)

2、Java串口编程

Java串口编程例子的更多相关文章

  1. Java串口编程学习2-读串口

    如果读串口出现乱码,则: 1.可能是波特率设置不对 2.可能是数据编码格式不对 import gnu.io.*; import java.awt.*; import java.awt.event.Ac ...

  2. Java串口编程学习1-环境配置(64位Win7)

    最近在做zigbee的课程设计,需要Java实现对串口数据的读写操作. 网上找了很多代码,好像都比较过时了,直接拿来用没法跑通……QAQ……然后自己写个教程留底,如有不当之处还请各位路过的大神赐教. ...

  3. java串口编程

    报错:no rxtxSerial in java.library.path thrown while loading gnu.io.RXTXCommDrive java.lang.Unsatisfie ...

  4. Java并发编程:volatile关键字解析

    Java并发编程:volatile关键字解析 volatile这个关键字可能很多朋友都听说过,或许也都用过.在Java 5之前,它是一个备受争议的关键字,因为在程序中使用它往往会导致出人意料的结果.在 ...

  5. JAVA并发编程J.U.C学习总结

    前言 学习了一段时间J.U.C,打算做个小结,个人感觉总结还是非常重要,要不然总感觉知识点零零散散的. 有错误也欢迎指正,大家共同进步: 另外,转载请注明链接,写篇文章不容易啊,http://www. ...

  6. Java并发编程:Thread类的使用

    Java并发编程:Thread类的使用 在前面2篇文章分别讲到了线程和进程的由来.以及如何在Java中怎么创建线程和进程.今天我们来学习一下Thread类,在学习Thread类之前,先介绍与线程相关知 ...

  7. Java并发编程:线程池的使用

    Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...

  8. Java并发编程:同步容器

    Java并发编程:同步容器 为了方便编写出线程安全的程序,Java里面提供了一些线程安全类和并发工具,比如:同步容器.并发容器.阻塞队列.Synchronizer(比如CountDownLatch). ...

  9. Java Web编程技术学习要点及方向

    学习编程技术要点及方向亮点: 传统学习编程技术落后,应跟著潮流,要对业务聚焦处理.要Jar, 不要War:以小为主,以简为宝,集堆而成.去繁取简 Spring Boot,明日之春(future of ...

随机推荐

  1. 第2.2节 Python的语句

    上节已经介绍了极简的Python代码编写,已经用到了赋值语句,本节对Python的程序语句进行介绍. 一. 常用命令 在介绍Python语句之前,先介绍一下几个有用的Python命令. dir(模块名 ...

  2. 老猿学5G:融合计费场景的离线计费会话的Nchf_OfflineOnlyCharging_Create创建操作

    ☞ ░ 前往老猿Python博文目录 ░ 一.Nchf_OfflineOnlyCharging_Create消息交互流程 Nchf_OfflineOnlyCharging_Create服务化操作请求是 ...

  3. Pentaho Report Designer 报表系统 - 入门详解

    目录 简介 安装与配置 环境要求 运行方式 使用教学 数据源配置与原始数据获取 报表布局设计与格式化 布局设计 模块结构 控件 示例 报表预览与发布 报表访问与获取 参考材料 简介 ​ Pentaho ...

  4. 常见SQL注入点判断

    sql注入手工检测 SQL注入手工检测 1基本检测 数字型 字符型 搜索型 POST注入 布尔盲注 报错注入 堆叠注入 判断是什么数据库 2绕过技巧 大小写 替换关键字 使用编码 注释和符号 等价函数 ...

  5. 熊海CMS xhcms v1.0代码审计

    有空的时候就进行小型CMS的代码审计,这次审计的对象是熊海CMS v1.0 本地环境安装好了之后,可以看到提示安装了锁文件 说明重装漏洞应该不会存在了,这时候丢进seay代码审计系统的代码也出结果了, ...

  6. 懒松鼠Flink-Boot(Flink+Spring):一款将Flink与Spring生态完美融合的脚手架工程

    目录 你可能面临如下苦恼: 接口缓存 重试机制 Bean校验 等等...... 它为流计算开发工程师解决了 有了它你的代码就像这样子: 仓库地址:懒松鼠Flink-Boot 1. 组织结构 2. 技术 ...

  7. centos安装scrapy

    安装scrapy centos 7 安装scrapy报错说找不到scrapy需要的Twisted13.0以上版本? Collecting Twisted>=13.1.0 (from Scrapy ...

  8. Angular:路由的配置、传参以及跳转

    ①路由的配置 1.首先用脚手架新建一个项目,在路由配置时选择yes 2.用ng g component创建组件 3.在src/app/app-routing.module.ts中配置路由 import ...

  9. Python条件判断和循环语句

    一.条件判断语句 通过一条或多条语句的判断来决定是否执行代码块 1.if语句基本形式: if 判断条件:    语句块 例如: score=75if score>=60:    print &q ...

  10. Docker的容器使用与连接-Window

    启动容器 启动容器之前需要先拉取镜像,然后通过 run 命令启动容器,同一个镜像可以启动多个容器,只要执行多次 run 命令就行了.我们这边启动 centos 的镜像. PS D:\> dock ...