序言
说到开源,恐怕非常少有人不挑大指称赞。学生通过开源码学到了知识,程序猿通过开源类库获得了别人的成功经验及可以按时完毕手头的project,商家通过开源软件赚到了钱……,总之是皆大欢喜。然而开源软件或类库的首要缺点就是大多缺乏具体的说明文档和使用的样例,或者就是软件代码随便你用,就是文档,样例和后期服务收钱。这也难怪,毕竟就像某个著名NBA球员说的那样:“我还要养家,所以千万美元下面的合同别找我谈,否则我宁可待业”。是啊,支持开源的人也要养家,收点钱也只是分。要想既不花钱又学到知识就仅仅能借助网络和了,我仅仅是想抛砖引玉,为开源事业做出点微薄共献,能为你的project解决哪怕一个小问题,也就足够了。
尽管我的这个系列介绍的东西不是什么Web框架,也不是什么开源server,可是我相信,作为一个程序猿,什么样的问题都会遇到。有时候越是简单的问题反而越棘手;越是小的地方就越是找不到称手的家伙。仅仅要你不是整天仅仅与“架构”、“构件”、“框架”打交道的话,相信我所说的东西你一定会用到。

1     串口通信简单介绍... 1
1.1      常见的Java串口包... 1
1.2      串口包的安装(Windows下)... 1
2     串口API概览... 2
2.1      javax.comm.CommPort2
2.2      javax.comm.CommPortIdentifier3
2.3      javax.comm.SerialPort3
2.4      串口API实例... 3
2.4.1       列举出本机全部可用串口... 3
2.4.2       串口參数的配置... 4
2.4.3       串口的读写... 4
3     串口通信的通用模式及其问题... 5
3.1      事件监听模型... 5
3.2      串口读数据的线程模型... 6
3.3      第三种方法... 7
4     结束语... 9

1     串口通信简单介绍
嵌入式系统或传感器网络的非常多应用和測试都须要通过PC机与嵌入式设备或传感器节点进行通信。当中,最经常使用的接口就是RS-232串口和并口(鉴于USB接口的复杂性以及不须要非常大的传输数据量,USB接口用在这里还是显得过于奢侈,况且眼下除了SUN有一个支持USB的包之外,我还没有看到其它直接支持USB的Java类库)。SUN的CommAPI分别提供了对经常使用的RS232串行port和IEEE1284并行port通讯的支持。RS-232-C(又称EIA RS-232-C,下面简称RS232)是在1970年由美国电子工业协会(EIA)联合贝尔系统、调制解调器厂家及计算机终端生产厂家共同制定的用于串行通讯的标准。RS232是一个全双工的通讯协议,它能够同一时候进行数据接收和发送的工作。
1.1 常见的Java串口包
眼下,常见的Java串口包有SUN在1998年公布的串口通信API:comm2.0.jar(Windows下)、comm3.0.jar(Linux/Solaris);IBM的串口通信API以及一个开源的实现。鉴于在Windows下SUN的API比較经常使用以及IBM的实现和SUN的在API层面都是一样的,那个开源的实现又不像两家大厂的产品那样让人放心,这里就仅仅介绍SUN的串口通信API在Windows平台下的使用。
1.2 串口包的安装(Windows下)
到SUN的站点下载javacomm20-win32.zip,包括的东西例如以下所看到的:
依照其使用说明(Readme.html)的说法,要想使用串口包进行串口通信,除了设置好环境变量之外,还要将win32com.dll拷贝到<JDK>/bin文件夹下;将comm.jar拷贝到<JDK>/lib;把javax.comm.properties也相同拷贝到<JDK>/lib文件夹下。然而在真正执行使用串口包的时候,仅作这些是不够的。由于通常当执行“java MyApp”的时候,是由JRE下的虚拟机启动MyApp的。而我们仅仅复制上述文件到JDK对应文件夹下,所以应用程序将会提示找不到串口。解决问题的方法非常easy,我们仅仅须将上面提到的文件放到JRE对应的文件夹下就能够了。
值得注意的是,在网络应用程序中使用串口API的时候,还会遇到其它更复杂问题。有兴趣的话,你能够查看CSDN社区中“关于网页上Appletjavacomm20读取client串口的问题”的帖子。
2     串口API概览
2.1      javax.comm.CommPort
这是用于描写叙述一个被底层系统支持的port的抽象类。它包括一些高层的IO控制方法,这些方法对于全部不同的通讯port来说是通用的。SerialPort 和ParallelPort都是它的子类,前者用于控制串行port而后者用于控这并口,二者对于各自底层的物理port都有不同的控制方法。这里我们仅仅关心SerialPort。
2.2      javax.comm.CommPortIdentifier
这个类主要用于对串口进行管理和设置,是对串口进行訪问控制的核心类。主要包含下面方法
l         确定是否有可用的通信port
l         为IO操作打开通信port
l         决定port的全部权
l         处理port全部权的争用
l         管理port全部权变化引发的事件(Event)
2.3        javax.comm.SerialPort
这个类用于描写叙述一个RS-232串行通信port的底层接口,它定义了串口通信所需的最小功能集。通过它,用户能够直接对串口进行读、写及设置工作。
2.4 串口API实例
大段的文字怎么也不如一个小样例来的清晰,以下我们就一起看一下串口包自带的样例---SerialDemo中的一小段代码来加深对串口API核心类的用法的认识。
2.4.1   列举出本机全部可用串口
void listPortChoices() {
CommPortIdentifier portId;
Enumeration en = CommPortIdentifier.getPortIdentifiers();
// iterate through the ports.
while (en.hasMoreElements()) {
portId = (CommPortIdentifier) en.nextElement();
if (portId.getPortType() == CommPortIdentifier.PORT_SERIAL) {
System.out.println(portId.getName());
}
}
portChoice.select(parameters.getPortName());
}
以上代码能够列举出当前系统全部可用的串口名称,我的机器上输出的结果是COM1和COM3。
2.4.2   串口參数的配置
串口一般有例如以下參数能够在该串口打开曾经配置进行配置:
包含波特率,输入/输出流控制,数据位数,停止位和齐偶校验。
SerialPort sPort;
try {
sPort.setSerialPortParams(BaudRate,Databits,Stopbits,Parity);
//设置输入/输出控制流
sPort.setFlowControlMode(FlowControlIn | FlowControlOut);
} catch (UnsupportedCommOperationException e) {}
2.4.3   串口的读写
对串口读写之前须要先打开一个串口:
CommPortIdentifier portId = CommPortIdentifier.getPortIdentifier(PortName);
try {
SerialPort  sPort = (SerialPort) portId.open("串口全部者名称", 超时等待时间);
} catch (PortInUseException e) {//假设port被占用就抛出这个异常
throw new SerialConnectionException(e.getMessage());
}
//用于对串口写数据
OutputStream os = new BufferedOutputStream(sPort.getOutputStream());
os.write(int data);
//用于从串口读数据
InputStream is = new BufferedInputStream(sPort.getInputStream());
int receivedData = is.read();
读出来的是int型,你能够把它转换成须要的其它类型。
这里要注意的是,由于Java语言没有无符号类型,即全部的类型都是带符号的,在由byte到int的时候应该尤其注意。由于假设byte的最高位是1,则转成int类型时将用1来占位。这样,原本是10000000的byte类型的数变成int型就成了1111111110000000,这是非常严重的问题,应该注意避免。
3     串口通信的通用模式及其问题
最终唠叨完我最讨厌的基础知识了,以下開始我们本次的重点--串口应用的研究。因为向串口写数据非常easy,所以这里我们仅仅关注于从串口读数据的情况。通常,串口通信应用程序有两种模式,一种是实现SerialPortEventListener接口,监听各种串口事件并作对应处理;还有一种就是建立一个独立的接收线程专门负责数据的接收。因为这两种方法在某些情况下存在非常严重的问题(至于什么问题这里先卖个关子J),所以我的实现是採用第三种方法来解决问题。
3.1 事件监听模型
如今我们来看看事件监听模型是怎样运作的
l        首先须要在你的port控制类(比如SManager)加上“implements SerialPortEventListener”
l        在初始化时添�例如以下代码:
try {
SerialPort sPort.addEventListener(SManager);
} catch (TooManyListenersException e) {
sPort.close();
throw new SerialConnectionException("too many listeners added");
}
sPort.notifyOnDataAvailable(true);
l        覆写public void serialEvent(SerialPortEvent e)方法,在当中对例如以下事件进行推断:
BI -通讯中断.
CD -载波检測.
CTS -清除发送.
DATA_AVAILABLE -有数据到达.
DSR -数据设备准备好.
FE -帧错误.
OE -溢位错误.
OUTPUT_BUFFER_EMPTY -输出缓冲区已清空.
PE -奇偶校验错.
RI - 振铃指示.
一般最经常使用的就是DATA_AVAILABLE--串口有数据到达事件。也就是说当串口有数据到达时,你能够在serialEvent中接收并处理所收到的数据。然而在我的实践中,遇到了一个十分严重的问题。
首先描写叙述一下我的实验:我的应用程序须要接收传感器节点从串口发回的查询数据,并将结果以图标的形式显示出来。串口设定的波特率是115200,川口每隔128毫秒返回一组数据(大约是30字节左右),周期(即持续时间)为31秒。实測的时候在一个周期内应该返回4900多个字节,而用事件监听模型我最多仅仅能收到不到1500字节,不知道这些字节都跑哪里去了,也不清楚究竟丢失的是那部分数据。值得注意的是,这是我将serialEvent()中全部处理代码都注掉,仅仅剩下打印代码所得的结果。数据丢失的如此严重是我所不能忍受的,于是我决定採用其它方法。
3.2 串口读数据的线程模型
这个模型顾名思义,就是将接收数据的操作写成一个线程的形式:
public void startReadingDataThread() {
Thread readDataProcess = new Thread(new Runnable() {
public void run() {
while (newData != -1) {
try {
newData = is.read();
System.out.println(newData);
//其它的处理过程
……….
} catch (IOException ex) {
System.err.println(ex);
return;
}
}
readDataProcess.start();
}
在我的应用程序中,我将收到的数据打包放到一个缓存中,然后启动还有一个线程从缓存中获取并处理数据。两个线程以生产者—消费者模式协同工作,数据的流向例如以下图所看到的:
 

这样,我就圆满攻克了丢数据问题。然而,没高兴多久我就又发现了一个相同严重的问题:尽管这回不再丢数据了,但是原本一个周期(31秒)之后,传感器节电已经停止传送数据了,但我的串口线程依旧在努力的执行读串口操作,在控制台也能够看见收到的数据仍在不断的打印。原来,因为传感器节点发送的数据过快,而我的接收线程处理只是来,所以InputStream就先把已到达却还没处理的字节缓存起来,于是就导致了明明传感器节点已经不再发数据了,而控制台却还能看见数据不断打印这一奇怪的现象。唯一值得庆幸的是最后收到数据确实是4900左右字节,没出现丢失现象。然而当处理完最后一个数据的时候已经快1分半钟了,这个时间远远大于节点执行周期。这一延迟对于一个实时的显示系统来说简直是灾难!
后来我想,是不是因为两个线程之间的同步和通信导致了数据接收缓慢呢?于是我在接收线程的代码中去掉了全部处理代码,仅保留打印收到数据的语句,结果依旧如故。看来并非线程间的通信阻碍了数据的接收速度,而是用线程模型导致了对于发送端数据发送速率过快的情况下的数据接收延迟。这里申明一点,就是对于数据发送速率不是如此快的情况下前面者两种模型应该还是好用的,仅仅是特殊情况还是应该特殊处理。
3.3 第三种方法
痛苦了许久(Boss天天催我L)之后,偶然的机会,我听说TinyOS中(又是开源的)有一部分是和我的应用程序相似的串口通信部分,于是我下载了它的1.x版的Java代码部分,參考了它的处理方法。解决这个问题的方法说穿了事实上非常easy,就是从根源入手。根源不就是接收线程导致的吗,那好,我就干脆取消接收线程和作为中介的共享缓存,而直接在处理线程中调用串口读数据的方法来解决这个问题(什么,为什么不把处理线程也一并取消?----都取消应用程序界面不就锁死了吗?所以必须保留)于是程序变成了这样:
public byte[] getPack(){
while (true) {
// PacketLength为数据包长度
byte[] msgPack = new byte[PacketLength];
for(int i = 0; i < PacketLength; i++){
if( (newData = is.read()) != -1){
msgPack[i] = (byte) newData;
System.out.println(msgPack[i]);
}
}
return msgPack;
}
}
在处理线程中调用这种方法返回所须要的数据序列并处理之,这样不但没有丢失数据的现象行出现,也没有数据接收延迟了。这里唯一须要注意的就是当串口停止发送数据或没有数据的时候is.read()一直都返回-1,假设一旦在開始接收数据的时候发现-1就不要理它,继续接收,直到收到真正的数据为止。

4     结束语

本文介绍了串口通信的基本知识,以及经常使用的几种模式。通过实践,提出了一些问题,并在最后加以解决。值得注意的是对于第一种方法,我曾将传感器发送的时间由128毫秒添加�到512毫秒,仍然有非常严重的数据丢失现象发生,所以假设你的应用程序须要非常精密的结果,数据传输的速率又非常快的话,就最好不要用第一种方法。对于另外一种方法,因为是线程导致的问题,所以对于不同的机器应该会有不同的表现,对于那些处理多线程比較好的机器来说,应该会好一些。可是我的机器是Inter 奔四3.0双核CPU+512DDR内存,这样都延迟这么厉害,还得多强的CPU才行啊?所以对于数据量比較大的传输来说,还是用第三种方法吧。只是这个世界问题是非常多的,并且未知的问题比已知的问题多的多,说不定还有什么其它问题存在,欢迎你通过以下的联系方式和我一起研究。

转载于:https://www.cnblogs.com/hrhguanli/p/3887401.html

Java串口通信具体解释相关推荐

  1. Java串口通信详解(转)

    Java串口通信详解(转) 作者:denimcc 日期:2007-05-11 序言     说到开源,恐怕很少有人不挑大指称赞.学生通过开源代码学到了知识,程序员通过开源类库获得了别人的成功经验及能够 ...

  2. Java串口通信报错# Problematic frame: # C [rxtxSerial.dll+0x4465](含详细解决流程)

    背景:在win10x64,myEclipse软件中移植他人的Java串口通信项目到自己电脑上时报错 问题:在GUI界面上点击"连接"按钮,GUI界面闪退,并在myEclipse软件 ...

  3. Java串口通信-JSerialComm

    Java串口通信-JSerialComm 目前网上的Java串口通信主要使用RXTXComm,但是这个库已经很久没有更新(最近的更新似乎在2012年),并且与JavaFX集成打包时会出现BUG.JSe ...

  4. java调用c 串口_基于C语言的java串口通信程序

    目录 1.前言 2.windows  串口通信API 3.C/C++封装  动态运行库 4.JAVA-JNI  java程序调用C++程序 一.前言 &ensp ;写这个博客主要是因为自己想用 ...

  5. java linux 串口_Linux Java 串口通信 | 学步园

    费了好大的劲搞定Linux系统上用Java写串口通信的问题. jdk中没有原生的串口api,网上找了半天的资料,大概知道了:Linux系统上用Java写串口程序,有两个包比较常用,一个是当年sun官方 ...

  6. Java串口通信详解

    序言 说到开源,恐怕很少有人不挑大指称赞.学生通过开源代码学到了知识,程序员通过开源类库获得了别人的成功经验及能够按时完成手头的工程,商家通过开源软件赚到了钱--,总之是皆大欢喜.然而开源软件或类库的 ...

  7. java串口通信DataRecive_串口通信之DataReceive事件触发时机

    环境:Windows PC.本机虚拟COM2连接COM3.串口调试助手COM2发数据 图1 1> ReceivedBytesThreshold为默认值1:2> 一次发送41个字节:3> ...

  8. java 串口 rxtx_【Java】基于RXTX的Java串口通信

    public classSerialPortManager {/*** 查找可用端口 * *@return可用端口名称列表*/ public static final ListfindPorts() ...

  9. Java串口通信读写串口导致程序崩溃问题

    最近心血来潮想用java实现与下位机的串口通讯,下位机使用STM32等间隔向串口发送八位的数据,上位机主要实现数据的接收和发送.过程中遇到的问题为:一旦执行到输入输出流对串口进行读写操作时,程序直接崩 ...

最新文章

  1. Xcode couldn‘t find any iOS App Development provisioning profiles matching ‘com.example.***‘
  2. java中ajax概念_Java之AJAX概念和实现方式
  3. 使用 .NET 实现 Ajax 长连接 (Part 1 - Comet Web Service)
  4. boost learn notes
  5. dmg文件转换iso ultraiso_mac去除dmg打开密码的方法
  6. 验证码实现php 难点,php实现简单的验证码功能
  7. boost::statechart模块实现延迟错误的测试程序
  8. QT中的模态对话框及非模态对话框
  9. osg图元绑定方式总结
  10. 解决Oracle 本地可以连接,远程不能连接问题
  11. 基于Udp的Socket网络编程
  12. JavaScript函数与对象
  13. Hadoop大数据组件简介
  14. 设为首页,加入收藏兼容360/火狐/谷歌/IE等主流浏览器的代码
  15. 大学计算机应用教程马秀麟,大学计算机基础电子教案.docx
  16. Python:体脂计算
  17. cad多段线画圆弧方向_CAD多段线绘制技巧 - CAD自学网
  18. latex formula
  19. 信息安全实验:实现一个fake-wifi
  20. 2022年乡村医生考试精选模拟题及答案

热门文章

  1. 用Fiddler调试localhost
  2. TWebBrowser的常见属性和方法
  3. 两个特征是独立好还是正相关好_【概率论与数理统计】第5期:随机变量的数字特征...
  4. 2-4实战分类之模型构建
  5. 计算机组装策划案,产品策划书格式
  6. java线程交替执行_Java synchronized线程交替运行实现过程详解
  7. easyconnect无法在mac上使用_Mac上Python无法输入中文- 2017年
  8. vba mysql 非法字符串_非法字符串处理.sql
  9. 拼接 结果集_JUST技术:利用轨迹拼接分析实时可达区域|技术前沿
  10. 中移4G模块-ML302-OpenCpu开发-服务器搭建