Java串口编程例子
最近笔者接触到串口编程,网上搜了些资料,顺便整理一下。网上都在推荐使用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,功能强大。
参考资料

Java串口编程例子的更多相关文章
- Java串口编程学习2-读串口
如果读串口出现乱码,则: 1.可能是波特率设置不对 2.可能是数据编码格式不对 import gnu.io.*; import java.awt.*; import java.awt.event.Ac ...
- Java串口编程学习1-环境配置(64位Win7)
最近在做zigbee的课程设计,需要Java实现对串口数据的读写操作. 网上找了很多代码,好像都比较过时了,直接拿来用没法跑通……QAQ……然后自己写个教程留底,如有不当之处还请各位路过的大神赐教. ...
- java串口编程
报错:no rxtxSerial in java.library.path thrown while loading gnu.io.RXTXCommDrive java.lang.Unsatisfie ...
- Java并发编程:volatile关键字解析
Java并发编程:volatile关键字解析 volatile这个关键字可能很多朋友都听说过,或许也都用过.在Java 5之前,它是一个备受争议的关键字,因为在程序中使用它往往会导致出人意料的结果.在 ...
- JAVA并发编程J.U.C学习总结
前言 学习了一段时间J.U.C,打算做个小结,个人感觉总结还是非常重要,要不然总感觉知识点零零散散的. 有错误也欢迎指正,大家共同进步: 另外,转载请注明链接,写篇文章不容易啊,http://www. ...
- Java并发编程:Thread类的使用
Java并发编程:Thread类的使用 在前面2篇文章分别讲到了线程和进程的由来.以及如何在Java中怎么创建线程和进程.今天我们来学习一下Thread类,在学习Thread类之前,先介绍与线程相关知 ...
- Java并发编程:线程池的使用
Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...
- Java并发编程:同步容器
Java并发编程:同步容器 为了方便编写出线程安全的程序,Java里面提供了一些线程安全类和并发工具,比如:同步容器.并发容器.阻塞队列.Synchronizer(比如CountDownLatch). ...
- Java Web编程技术学习要点及方向
学习编程技术要点及方向亮点: 传统学习编程技术落后,应跟著潮流,要对业务聚焦处理.要Jar, 不要War:以小为主,以简为宝,集堆而成.去繁取简 Spring Boot,明日之春(future of ...
随机推荐
- 译文:二进制序列类型 --- bytes, bytearray
在进行一些内置函数调用时,会发现bytes类型的参数或返回值,这个类型老猿前面没有介绍过,在此就不单独介绍了,直接从Python官网的内容用翻译软件翻译过来稍微修改. 操作二进制数据的核心内置类型是 ...
- PyQt(Python+Qt)学习随笔:QAbstractScrollArea的sizeAdjustPolicy、horizontalScrollBarPolicy、verticalScrollB属性
老猿Python博文目录 老猿Python博客地址 Qt Designer中QAbstractScrollArea包括三个属性,分别是horizontalScrollBarPolicy.vertica ...
- Djang项目部署之sqlite版本升级
项目环境: centos7 django 2.2.10 问题描述: 使用了django 2.2.12版本开发项目,此版本对应的sqlite需要升级为3.8.0以上. 百度了不少解决方案,缺点:过程繁琐 ...
- Metasploit魔鬼训练营第一章作业
1, Samba服务 Samba是在Linux和UNIX系统上实现SMB协议的一个免费软件,由服务器及客户端程序构成.SMB(Server Messages Block,信息服务块)是一种在局域网上共 ...
- Day4 【Scrum 冲刺博客】
每日会议总结 昨天已完成的工作 方晓莹(PIPIYing) 完善人员管理页的未完成部分 方子茵(Laa-L):无 黄芯悦(Sheaxx) 开始投诉反馈页面的开发 舒雯钰(LittleTaro) 博客的 ...
- Mysql为什么使用b+树,而不是b树、AVL树或红黑树?
首先,我们应该考虑一个问题,数据库在磁盘中是怎样存储的?(答案写在下一篇文章中) b树.b+树.AVL树.红黑树的区别很大.虽然都可以提高搜索性能,但是作用方式不同. 通常文件和数据库都存储在磁盘,如 ...
- win32 C++制作美观按钮,告别win32 API编程中默认的灰色按钮
使用win32 API制作美观按钮,当鼠标移入/移出按钮时改变按钮背景颜色,类似HTML网页中的效果,告别win32 API编程中默认的灰色按钮,效果图见下面动图和视频. 下载地址: 按钮效果(win ...
- monkey在指定的activity里面运行
下载包地址:链接: https://pan.baidu.com/s/1Wk2eOj3saZx71Mx6pT2L4Q 提取码: gupa 运行方式:步骤1: 将工具下载下来放到本地目录下,解压步骤2:配 ...
- IOS开发中设置导航栏主题
/** * 系统在第一次使用这个类的时候调用(1个类只会调用一次) */ + (void)initialize { // 设置导航栏主题 UINavigationBar *navBar = [UINa ...
- Golang之应用测试
Go 应用测试 测试的覆盖率 命令: go test ./ -v -cover 在<Go Web 编程>一书中,有以下结论: 这并不是绝对的,测试文件可以在不同的包,进行测试也是不会出现问 ...