Java串口通信详解_第1页
Java串口通信详解_第2页
Java串口通信详解_第3页
Java串口通信详解_第4页
Java串口通信详解_第5页
已阅读5页,还剩28页未读 继续免费阅读

付费下载

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

♦Java串口通信详解

序言

说到开源,恐怕很少有人不挑大指称赞。学生通过开源代码学到了知识,程序员通过开源

类库获得了别人的成功经验及能够按时完成手头的工程,商家通过开源软件赚到了钱,总之

是皆大欢喜。然而开源软件或类库的首要缺点就是大多缺乏详细的说明文档和使用的例子,或者

就是软件代码随便你用,就是文档,例子和后期服务收钱。这也养怪,毕竟就像某个著名NBA

球员说的那样:''我还要养家,所以千万美元以下的合同别找我谈,否则我宁可待业是啊,支

持开源的人也要养家,收点钱也不过分。要想既不花钱乂学到知识就只能借助网络和了,我只是

想抛砖引玉,为开源事业做出点微薄共献,能为你的工程解决哪怕一个小问题,也就足够了。

虽然我的这个系列介绍的东西不是什么Web框架,也不是什么开源服务器,但是我相信,

作为一个程序员,什么样的问题都会遇到。有时候越是简单的问题反而越棘手;越是小的地方就

越是找不到称手的家伙。只要你不是整天只与''架构〃、'、构件框架"打交道的话,相信我所说

的东西你一定会用到。

1串口通信简介

1.1常见的Java串口包

1.2串口包的安装(Windows下)

2串口API概览

2.1m.CommPort

2.2m.CommPortldentifier

2.3m.SerialPort

2.4串口API实例

2.4.1列举出本机所有可用串口

2.4.2串口参数的配置

2.4.3串口的读写

3串口通信的通用模式及其问题

3.1事件监听模型

3.2串口读数据的线程模型

3.3第三种方法

4结束语

1串口通信简介

抿入式系统或传感器网络的很多应用和测试都需要通过PC机与嵌入式设备或传感器节点

进行通信。其中,最常用的接口就是RS-232串口和并口(鉴于USB接口的复杂性以及不需要

很大的数据传输量,USB接口用在这里还是显得过于奢侈,况且目前除了SUN有一个支持USB

的包之外,我还没有看到其他直接支持USB的Java类库)。SUN的CommAPI分别提供了

对常用的RS232串行端口和IEEE1284并行端口通讯的支持。RS-232-C(又称EIA

RS-232-C,以卜.简称RS232)是在1970年由美国电子工业协会(EIA)联合贝尔系统、调制解

调器厂家及计算机终端生产厂家共同制定的用于串行通讯的标准。RS232是•个全双工的通讯

协议,它可以同时进行数据接收和发送的工作。

1.1常见的Java串口包

目前,常见的Java串口包有SJN在1998年发布的串口通信API:comm2.0.jar(Windows

下)、comm3.0.jar(Iinux/Solaris);IRM的串口通信API以及一个开源的实现。鉴于在

Windows下SUN的API比较常用以及IBM的实现和SUN的在API层面都是一样的,那个开

源的实现又不像两家大厂的产品那样让人放心,这里就只介绍SUN的串口通信API在Windows

平台下的使用。

1.2串口包的安装(Windows下)

到SUN的网站下载javacomm20-win32.zip,包含的东西如下所示:

文件9—帝令©收藏荚⑥)选项如帮助也)

等因嗡出恸时钳

:添加解压到测试查看删除查找向导

[王I::.javacomm20wiit32.zipVcommapi-ZIP压缩文件,解包大<1

名称O大小压纣

0..

liavadocs

L1samples

国comm.j^>28,043

[■apichanges.html3,335

/C0MM2.0_license.txt8,141

/CommAPI_FAQ.txt5,374

javax.comm,proper"也〉467

MdkL2.html2,182

^JPlatformSpecific.html3,715

©Readme.html3,913

固ReceiveBehavior.html1,821

com.dll^>27,648

按照其使用说明(Readme.html)的说法,要想使用串11包进行串11通信,除了设置好环

境变量之外,还要将win32com.dll复制到vJDK>\bin目录下:将comm.jar复制到

<JDK>\lib:把javax.8mm.properties也同样拷贝到vJDK>\lib目录下。然而在真正运行

使用串口包的时候,仅作这些是不够的。因为通常当运行''javaMyApp"的时候,是由JRE下

的虚拟机启动MyApp的。而我们只复制上述文件到JDK相应目录下,所以应用程序将会提示

找不到串口。解决这个问题的方法很简单,我们只须将上面提到的文件放到JRE相应的目录下

就可以了。

值得注意的是,在网络应用程序中使用串口API的时候,还会遇到其他更复杂问题。有兴

趣的话,你可以查看CSDN社区中''关于网页上Applet用javacomm20读取客户端串口的

问题”的帖子。

2串口API概览

2.1m.CommPort

这是用于描述一个被底层系统支持的端口的抽象类。它包含一些高层的10控制方法,这些方

法对于所有不同的通讯端口来说是通用的。SerialPort和ParallelPort都是它的子类,前者用

丁控制串行端口而后者用于控这并口,二者对于各自底层的物理端口都有不同的控制方法。这里

我们只关心SerialPort;.

2.2m.CommPortldentifier

这个类主要用于对串口进行管理和设置,是对串口进行访问控制的核心类。主要包括以下方法

|确定是否有可用的通信端口

|为io操作打开通信端口

|决定端口的所有权

।处理端口所有权的争用

I管理端口所有权变化引发的事件(Event)

2.3m.SerialPort

这个类用于描述一个RS-232串行通信端I」的底层接口,它定义了串口通信所需的最小功

能集。通过它,用户可以直接对串口进行读、写及设置工作。

2,4串口API实例

大段的文字怎么也不如一个小例子来的清晰,下面我们就一起看一下串【」包自带的例子

—SerialDemo中的一小段代码来加深对串口API核心类的使用方法的认识。

2.4.1列举出本机所有可用串口

voidlistPortChoices(){

CommPortldentifierportld;

Enumerationen=CommPortIdentifier.getPortIdentifiers();

//iteratethrojghtheports.

while(en.hasMoreElements()){

portld=(CommPortldentifier)en.nextElement();

if(portId.getPortType()==CommPortldentifier.PORT_SERIAL){

System.out.println(portId.getName());

)

}

portchoice.select(parameters.getPortName());

)

以上代码可以列举出当前系统所有可用的串口名称,我的机器上输出的结果是COM1和

COM3。

2.4.2串口参数的配置

串口一般有如下参数可以在该串口打开以前配置进行配置:

区]

包括波特率,输入/输出流控制,数据位数,停止位和齐偶校睑。

SerialPortsPort;

try{

sPort.setSerialPortParams(BaudRate,Databits,Stopbits,Parity);

〃设置输入/输出控制流

sPort.setFlowControlMode(FlowContrclIn|FlowControlOut);

}catch(UnsupportedCommOperationExceptione){}

2.4.3串口的读写

对串口读写之前褥要先打开一个串口:

CommPortldentifierportld=CommPortldentlfler.getPortldentlfler(PortName);

try(

SerialPortsPort=(SerialPort)portld.open("串口所有者名称",超时等待

时间);

}catch(PortlnUseExceptione){〃如果端11被占用就抛出这个异常

thrownewSerialConnectionException(e.getMessage());

}

〃用于对串口写数据

Outputstreamos=newBufferedOutputStream(sPort.getOutputStream());

os.write(intdata);

〃用于从串口读数据

Inputstreamis=newBufferedlnputStream(sPort.getlnputStreamO);

intreceivedData=is.read();

读出来的是int型,你可以把它转换成需要的其他类型。

这里要注意的是,由于Java语言没有无符号类型,即所有的类型都是带符号的,在由byte

到int的时候应该尤其注意。因为如果byte的最高位是1,则转成int类型时将用1来占位。

这样,原本是10000000的byte类型的数变成int型就成了1111111110000000,这是很

严重的问题,应该注意避免。

3串口通信的通用模式及其问题

终于唠叨完我最讨厌的基础知识了,卜而开始我们本次的重点--串口应用的研究。由于向串

口写数据很简单,所以这里我们只关注于从串口读数据的情况。通常,串口通信应用程序有两种

模式,一种是实现SerialPortEventListener接口,监听各种串口事件并作相应处理;另一种

就是建立•个独立的接收线程专门负责数据的接收。由于这两种方法在某些情况下存在很严重的

问题(至于什么问题这里先卖个关子J),所以我的实现是采用笫三种方法来解决这个问题。

3.1事件监听模型

现在我们来看看事件监听模型是如何运作的

首先需要在你的端11控制类(例如SManager)加上''implementsSerialPortEventListener,z

在初始化时加入如下代码:

try(

ScrialPortsPo-t.addEvcntListcncr(SManagcr);

}catch(TooMan/ListenersExceptione){

sPort.close();

thrownewSerialConnectionException("toomanylistenersadded");

)

sPort.notifyOnDataAvailable(true);

覆写publicvoidserialEventiSerialPortEvente)方法,在其中对如下事件进行判断:

BI-通讯中断.

CD-载波检测.

CTS-清除发送.

DATA_AVAILABLE-有数据到达.

DSR短据设备准备好.

FE帧错误.

0E-溢位错误.

OUTPUT_BUFFER_EMPTY-输出缓冲区已清空.

PE-奇偶我验错.

RI-振铃指示.

一般最常用的就是DATA_AVAILABLE—串口有数据到达事件。也就是说当串口有数据到

达时,你可以在serialEvent中接收并处理所收到的数据。然而在我的实践中,遇到了一个十

分严重的问题。

首先描述一下我的实验:我的应用程序需要接收传感器节点从串口发回的查询数据,并将

结果以图标的形式显示出来。串口设定的波特率是115200,川口每隔128亳秒返回一组数据

(大约是30字节左右),周期(即持续时间)为31秒。实测的时候在一个周期内应该返回4900

多个字而用事件监听模型我最多只能收到不到1500字节,不知道这些字节都跑哪里去了,

也不清楚到底丢失的是那部分数据。值得注意的是,这是我将serialEvent。中所有处理代码都

注掉,只剩下打印代码所得的结果。数据丢失的如此严重是我所不能忍受的,于是我决定采用其

他方法。

3.2串口读数据的线程模型

这个模型顾名思义,就是将接收数据的操作写成一个线程的形式:

publicvoidstartReadingDataThread(){

ThreadreadDataProcess=newThread(newRunnable(){

publicvoidrun(){

while(newData!=-1){

try(

newData=is.read();

System,out.println(nAwData);

〃其他的处理过程

}catch(lOExceptionex){

System.err.println(ex);

return;

}

}

readDataProcess.start();

)

在我的应用程序中,我将收到的数据打包放到一个缓存中,然后启动另一个线程从缓存中

获取并处理数据。两个线程以牛.产者一消费者模式协同工作,数据的流向如下图所示:

这样,我就圆满解决了丢数据问题。然而,没高兴多久我就又发现了一个同样严重的问题:

虽然这回不再丢数据了,可是原本一个周期(31秒)之后,传感器节电已经停止传送数据了,

但我的串口线程依然在努力的执行读串口操作,在控制台也可以看见收到的数据仍在不断的打

印。原来,由于传感器由点发送的数据过快,而我的接收线程处理不过来,所以InputStream

就先把已到达却还没处理的字节缓存起来,于是就导致了明明传感器节点已经不再发数据了,而

控制台却还能看见数据不断打臼这一奇怪的现象。唯一值得庆幸的是最后收到数据确实是4900

左右字节,没出现丢失现象。然而当处理完最后一个数据的时候已经快1分半钟了,这个时间

远远大于节点运行周期。这一延迟对于一个实时的显示系统来说简直是灾难!

后来我想,是不是由于两个线程之间的同步和通信导致了数据接收缓慢呢?于是我在接收

线程的代码中去掠了所有处理代码,仅保留打印收到数据的语句,结果依然如故。看来并不是线

程间的通信阻碍了数据的接收速度,而是用线程模型导致了对于发送端数据发送速率过快的情况

下的数据接收延迟。这里申明一点,就是对于数据发送速率不是如此快的情况下前面者两种模型

应该还是好用的,只是特殊情况还是应该特殊处理。

3.3第三种方法

痛苦了许久(Boss天天催我L)之后,偶然的机会,我听说TinyOS中(又是开源的)有

一部分是和我的应用程序类似的串口通信部分,于是我下载了它的1.x版的Java代码部分,参

考了它的处理方法。解决问题的方法说穿了其实很简单,就是从根源入手。根源不就是接收线程

导致的吗,那好,我就干脆取消接收线程和作为中介的共享缓存,而直接在处理线程中调用串口

读数据的方法来解决问题(什么,为什么不把处理线程也一并取消?----都取消应用程序界面不

就锁死了吗?所以必须保留)于是程序变成了这样:

publicbyte[]getPack(]<

while(true){

//PacketLength为数据包长度

byte[]msgPack=newbyte[PacketLength];

for(inti=0;i<PacketLength;i++){

if((newData=is.read())!=-1){

msgPack[i]=(byte)newData;

System.out.pnntln(msgPack[ij);

}

)

returnmsgPack;

}

}

在处理线程中调用这个方法返回所需要的数据序列并处理之,这样不但没有丢失数据的现

象行出现,也没有数据接收延迟了。这里唯一需要注意的就是当串口停止发送数据或没有数据的

时候is.read。一直都返回-1,如果一旦在开始接收数据的时候发现-1就不要理它,继续接收,

直到收到真正的数据为止。

4结束语

本文介绍了串口通信的基本知识,以及常用的几种模式。通过实践,提出了一些问题,并在

最后加以解决。值得注意的是对于第一种方法,我曾将传感器发送的时间由128亳秒增加

到512亳秒,仍然有很严重的数据丢失现象发生,所以如果你的应用程序需要很精密的结

果,传输数据的速率又很快的话,就最好不要用第一种方法。对于第二种方法,由于是线程

导致的问题,所以对于不同的机器应该会有不同的表现,对于那些处理多线程比较好的机器

来说,应该会好一些。但是我的机器是Inter奔四3.0双核CPU+512DDR内存,这样都

延迟这么厉害,还得多强的CPU才行啊?所以对于数据量比较大的传愉来说,还是用第三

种方法吧。不过这个世界问题是很多的,而且未知的问题比已知的问题多的多,说不定还有

什么其他问题存在,欢迎你通过下面的联系方式和我一起研究.

关于java使用javacomm20-win32实践总结

山于这几天要通过java国用通过聿口或井口连接的硬件资源,所以我就要用到和底层的

硬件进行通讯。通过RS-232的通讯协议,了解电脑和外设是怎样进行通讯的。在应用中我们也

可以通过JNI来实现(详情请见/blog/31508),这样的话,就必须

知道更多的知识。由于java已经提供我们一个javacomm20-win32通用的API我们还是实行"拿

来主义”吧。我就把整个应用的过程详细的说一下,希望给需要的人一点帮助。

我们经过串口和外设通讯,下面我就以串口为例进行解说。

1)我们要准备相应的设备。

电脑,外设,通过数据线把他们连接起来。

2)检验外设到底是用的那个COM口和电脑通讯的.

也就是说,他们有没有真确的连接上。我们可以通过下载串口通讯口测试软件,我用

的是"$叩62(^[7110。166”的绿色软件,进行测试的.这软件很适应,如果选中的某个COM已

经被使用了,它会给你一个相应的提示(端口以被占用)。如果你不知道到底是使用的那个端口,

那么你可以通过superCommTool软件个一个的试,如果正常的话,那么你可以看到有数据显

示在数据接收窗口。也许,有些主板的串口坏了,那么你就要买一个转接卡,通过PCI插口转

接。

3)察看外设使用说明书知道外设的相关参数.

比如,波特率,数据位,停止位,校验位,等等。只有正确参数,才能显示正确的数

据。当然,你可以在通讯测试软件上调试这些参数的。比如:波特率=2400,数据位=8,

停止位=2,校验位=1»

4)准备开发环境。

最基本的JDK了,你可以使用自己钟爱的IDE,帮助你开发。IDE可能自带了JDK,

那么你要把相应的javaComm20-win32放到运行时使用的JDK中。下载JAVAcomm20-win32。

5)了解javaComm20-win32o

你必须把win32com.dll复制到java.home/bin下:Jt!perties复制到

java.home/lib下:把comm.jar添加到你classPath下。前面两个都是非常重要的。

下面说明用到的几个类:

m.CDmmPortldentifier

通讯端口管理器,CommPortldentifier是控制访问到通讯端口的中心类。它包

括的方法有:

a.通过驱动决定通讯端口是可用的。

b.打开通讯端口为了I/O操作。

c.决定端门的拥有者。

d.解析端口拥有者的争夺。

e.管理事件显示在端口拥有者的中的状态改变。

一个应用程序首先使用CommPortldentifier中的方法,通过相关的驱动去获取那

些通讯端口是可用的并且选择一个端口便于开始。然后它使用方法在其它类中

想CommPort,ParallelPort和SerialPort通过这个端口进行通讯。

m.SerialPort

一个RS-232串口通讯端口。SerialPort描述底层的接口到一个串口通讯端

变得有效的通过底层的系统。SerialPort定义最小的必需的功能便于串口通讯端口。

m.SerialPortEventListener串行端口事件传播。

m.CommDriver

6)代码的编写。

a.获取SerialPortsPort对象的两种方法。

1)2)

java代码

1.System.loadLibrary("win32com");

2.m.CommDriverdriver=null;

3.StringdriverName="m.Win32Dri

ver";

4.SerialPortsPort=(SerialPort)driver.getCommP

ort("COM4",ommPortIdentifier.PORT_SERIAL);

java代码

1.CommPortldentifierportld=CommPortldentifier

.getPortIdentifier("COM4");

2.SerialPortsPort=(SerialPort)portId.open("ship

menf/lOOO);

以上两种方法都可以。不过一般都会采用第二种,方法说明我们获取了对串行

端口(COM4),可以和它进行通讯了。

b.设置串行端口通讯参数。

java代码

1.sPort.setSerialPortParams(2400zSerialPort.DATA

BITS_8zSerialPort.STOPBITS_2,SerialPort.PARIT

Y_NONE);

c.获取输入(出)流。

java代码

1.Inputstreamis=sPortgetInputStream();〃从外'

设获取数据

2.Outputstreamos=sPort.getOutputStream();//

发送命令到外设

d.通过监听器就可以得到数据了。

java代码

1.//SetnotifyOnDataAvailabletotruetoalloweve

ntdriveninput.

2.sPort.notifyOnDataAvailable(true);

3.

4.//SetnotifyOnBreaklnterruptoallowe

ventdrivenbreakhandling.

5.sPort.notifyOnBreaklnterrupt(true);

6.

7.//Setreceivetimeouttoallowbreaking

outofpollingloopduringinputhandling.

8.sPort.enableReceiveTimeout(30);

9.StringBufferlinkWgt=newStringBuffer

。;〃存放获取的数据

10.sPort.addEventListener(

11.newSerialPortEventListener(){

12.publicvoidserialEvent(SerialPortE

vente){

13.intnewData=0;

14.//Determinetypeofevent.

15.switch(e.getEventType()){

16.//Readdatauntil-1isreturne

d.If\risreceivedsubstitute

17.//\nforcorrectnewlinehan

dling.

18.caseSerialPortEvent.DATA_

AVAILABLE:

19.while(newData!=-1){

20.try{

21.newData=is.read()

/

22.if(newData==-1)

{

23.break;

24.}

25.if('\r'==(char)new

Data){

26.}else{

27.linkWgt.append((

char)newData);

28.}

29.}catch(lOExceptione

x){

30.System.err.println(e

x);

31.return;

32.}

33.}

34.

35.//Appendreceiveddatato

messageArealn.

36.

37.try(

38.System.out.println("linkW

gt---------HUI"+Double.valueOf(linkWg

t.toString()));

39.

40.}catch(Exceptionew){

41.ew.printStackTrace();

42.}finally{

43.try{

44.〃用完了,记得关闭端

口。

45.is.close();

46.sPort.close();

47.}catch(Exceptionc){

48.c.printStackTrace();

49.}

50.}

51.break;

52.11IfbreakeventappendBR

EAKRECEIVEDmessage.

53.caseSerialPortEvent.BI:

54.System.out.println("\n-

BREAKRECEIVED—\n");

55.)

56.)

57.

58.}

59.);

7)常见的异常

a.m.NoSuchPortException这个说明你的perties

没有放到正确的位置。

如果有什么不正确的地方,欢迎批评指正,谢谢!

Java串行端口通讯技术

了解串行通讯

串行通讯协议有很多种,像RS232,RS485,RS422,甚至现今流行的USB等都是串行

通讯协议。而串行通讯技术的应用无处不在。可能大家见的最多就是电脑的串口与Modem的

通讯。记得在PC机刚开始在中国流行起来时(大约是在90年代前五年),那时甚至有人用一

条串行线进行两台电脑之间的数据共享。除了这些,手机,PDA.USB鼠标、键盘等等都是以

串行通讯的方式与电脑连接。而笔者工作性质的关系,所接触到的就更多了,像多串口卡,各种

种类的具有串口通讯接口的检洌与测量仪器,串口通讯的网络设备等。

虽然串行通讯有很多种,但笔者所知的在整个电广通讯产品方面,以RS232的通讯方式最

为多见。虽然USB接口的电子产品也是层出不穷,但了解一下Java在串行通讯方面的技术还

有有必要的,说不定有哪位读者还想用此技术写一个PDA与电脑之间数据共享的程序呢。

本文主要以RS232为主来力解JAVA的串行通讯技术。

RS232通讯基础

RS-232-C(又称EIARS-232-C,以下简称RS232)是在1970年由美国电子工业协会

(EIA)联合贝尔系统、调制解调器厂家及计算机终端生产厂家共同制定的用于串行通讯的标准。

RS232是一个全双工的通讯协议,它可以同时进行数据接收和发送的工作。RS232的端口通

常有两种:9针(DB9)和25针(DB25)。

DB9和DB25的常用针脚定义

9针串口(DB9)25针串口(DB25)

针号功能说明缩写针令功能说明缩写

1数据载波检测DCD8数据载波检测DCD

2接收数据RXD3接收数据RXD

3发送数据TXD2发送数据TXD

4数据终端准备DTR20数据终端准备DTR

5信号地GND7信号地GND

6

6数据设备准备好DSR数据准备好DSR

7请求发送RTS4请求发送RTS

8清除发送CTS5清除发送CTS

9振铃指示RI22振铃指示RI

常见的边线方式

常见的通讯方式是三线式,这种方式是将两个RS232设备的发送端(TXD)和接收端(RXD)及接

地端(GND)互相连接,也是许多读者所知道的连接方式:

(9针)

F\AL/)____1AL>

3(TXD)—-----2(TXD)

5(GND)—­......5(GND)

(25针)

2(RXD)—-----3(TXD

3(TXD)—-----2(RXD)

7(GND)—-----7(GND)

这种方式分别将两端的RS232接口的2--3,3---2,5(7)---5(7)针脚连接起来。其中2是数

据接收线(RXD),3是数据发送线(TXD),5(7)是接地(RND)。如果有一台式PC,和部

NoteBook电脑,就可以用这种方式连线了。用三线式可以将大多数的RS232设备连接起来。

但如果你认死了2--3,3・・2,5(7)-5(7)对接这个理,会发现在连某些RS232设备时并不奏效。

这是因为有些设备在电路内部已将2和3线调换过来了,你只要2,3,5(7)针••对应就行了。

小技巧:如何辨别TXD和RXD端口?

搞电子的人手边应该常备一个电表,用来测测电压,电阻什么的会很有用。你

只要分别测一下RS232端口的2-5或3-5针脚之间的电压,通常TXD针脚与GND

之间会有3~15V左右的负电压,表示它是TXD针脚。

安装JavaCommunicationsAPI

Sun的J2SE中并没有直接提供以上提到的任何一种串行通讯协议的开发包,而是以独立的

jar包形式发布在java.sun.ccm网站上(从这里下载)—即comm.jar,称之为Javatm

CommunicationsAPL它是J2SE的标准扩展。comm.jar"并不是最近才有,早在1998年

时,sun就已经发布了这个开发包。comm.jar分别提供了对常月的RS232串行端口和

IEEE1284并行端口通讯的支持。目前sun发布的comm.jar只有Windows和Solaris平台

两个版本,如果你需要Linux平台下的,可以在

/~kevinh/linuxcomm.html找到。

在使用8mm.jar之前,必须知道如何安装它。这也是困扰许多初学javaRS232通讯者

的一个难题。如果我们电脑上安装了JDK,它将同时为我们安装一份JRE(JavaRuntime

Entironment),通常我们运行程序时都是以JRE来运行的。所以以下的安装适用于JRE。如果

你是用JDK来运行程序的,请格相应的改成。

下载了comm.jar开发包后,与之一一起的还有两个重要的文件,win32com.dll和

pertiesocomm.jar提供了通讯用的javaA>L而win32com.dll提供了

供comm.jar调用的本地驳动登口。而perties是这个驱动的类配置文件。

首先将comm.jar复制到\lib\ext目录。再将win21com.dll复制到你的RS232应用程序运

行的目录,即user.dk然后将perties货制到\lib目录。

通讯前的准备

如果你手头上没有现成的提供了标准RS232串11的设备,你可以将自己的电脑模拟成两台

不同的串口设备。通常电脑主机后面的面板提供了两个9针的串口,请将这两个串口的2,3,5

脚按前面介绍的方法连接。电子市场都有现成的连接头卖,请不要买那种封装的严严实实的接头,

而要买用螺丝封装可以拆开的连接头,这样可以方便自己根据需要连接各个针脚。

CommAPI基础

我无意于在此详细描述CommAPI每个类和接口的用法,但我会介绍CommAPI的类结

构和几个重要的API用法。

所有的commAPI位于m包下而。从CommAPI的javadoc来看,它介绍

给我们的只有区区以下13个类或接口:

m.CommDriver

m.CommPort

m.ParallelPort

javax.comm.SerialPort

m.CommPortldentifier

javax.comm.CommPortOwnershipListener

javax.comm.ParallelPortEvent

m.SerialPortEvent

javax.comm.ParallelPortEventListener(extendsjava.util.EventListener)

javax.comm.SerialPortEventListener(extendsjava.util.E/entListener)

jam.NoSuchPortException

m.PortinUseException

m.UnsupportedCommOperationException

下面讲解一下几个主要类或接口。

1.枚举出系统所有的RS232端口

在开始使用RS232端口通水之前,我们想知道系统有哪些端口是可用的,以卜.代码列出系

统中所有可用的RS232端口:

Enumerationcn=CommPortldentifier.gctPortIdcntificrs();

CommPortldentifierportld;

while(en.hasMoreElements())

{

portld=(CommPortldentifier)en.nextElement();

/*如果端口类型是串口,则打印出其端口信息*/

if(portld.getPortType()==CommPortldentifier.PORT_SERIAL)

{

System.out.println(portId.getName());

}

}

在我的电脑上以上程序输出以下结果:

COM1

COM2

CommPortldentifier类的getPortldentifiers方法可以找到系统所有的串口,每个串口对

应一个CommPortldentifier类的实例。

2.打开端口

如果你使用端II,必须先打开它。

try{

CommPortserialPort=portld.open("MyApp",60);

/*从端口中读取数据*/

Inputstreaminput=serialPort.getInputStream();

input.read(...);

/*往端口中写数据*/

Outputstreamoutput=serialPort.getOutputStream();

output.write(...)

}catch(PortInUseExceptionex)

{...}

通过CommPortIdentifieropen方法可以返回•个CommPort对象。open方法有两

个参数,第一个是String,通常设置为你的应用程序的名字。第二个参数是时间,即开启端口

超时的亳秒数0当端II被另外的应用程序占用时,将抛出PortlnUseException异常。

在这里CommPortIdentifier类和CommPort类有什么区别呢?其实它们两者是一一对应

的关系。CommPortldentifie「主要负责端口的初始化和开启,以及管理它们的占有权。而

CommPort则是跟实际的输入和输出功能有关的。通过CommPort的getInputStream()«J

以取得端口的输入流,它是java.io.InputStream接口的一个实例。我们可以用标准的

Inputstream的操作接口来读取流中的数据,就像通过Fileinputsteam读取文件的内容一样。

相应的,CommPort的getOLtputStream可以获得端口的输出流,这样就可以往串口输出数

据了。

3.关闭端口

使用完的端口,必须记得将其关闭,这样可以让其它的程序有机会使用它,不然其它程序使

用该端口时可能会抛出端口正在使用中的错误。很奇怪的是,CommPortIdentifier■类只提供

了开启端口的方法,而要关闭端口,则要调用CommPort类的dose。方法。

通讯方式

CommPort的输入流的读取方式与文件的输入流有些不一样,那就是你可能永远不知这个

Inputstream何时结束,除非对方的Outputstream向你发送了一个特定数据表示发送结束,

你收到这个特定字符后,再行关闭你的InputStream=而comm.jar提供了两种灭活的方式让

你读取数据。

1.轮询方式(Polling)

举个例子,你同GF相约一起出门去看电影,但你的GF好打扮,这一打扮可能就是半小时

甚至一小时以上。这时你就耐不住了,每两分钟就催问一次''好了没?",如此这样,直到你的

GF说0K了才算完。这个就叫轮询(Polling)。

在程序中,轮询通常设计成一个封闭的循环,当满足某个条件时即结束循环。刚才那个例子

中,你的GF说''0K了!〃,这个就是结束你轮询的条件。在单线程的程序中,当循环一口执行

某项任务而又无法预知它何时结束时,此时你的程序看起来可能就像死机一样。在VB程序中,

这个问题可以用在循环结构中插入一个doEvent语句来解决。而Java中,最好的方式是使用

线程,就像以下代码片断一样。

publicTestPortextendThread

{

Inputstreaminput=serialPort.getInputStream();

StringBufferbuf=newStringBuffer();

booleanstopped=false;

publicvoidrun()

{

try(

while(!stopped)

intch=input.read():

if(ch=='q'||ch=='Q')

{

/*结束读取,关闭端口…*/

stopped=true;

}

else

<

buf.append((char)ch);

}catch(InterruptedExceptione){}

}

2.监听方式(listening)

CommAPI支持标准的JavaBean型的事件模型。也就是说,你可以使用类似

AddXXXListener这样的方法为•个串口注册自己的监听器,以监听方式进行数据读取。

如要对端口监听,你必须先取得CommPortldentifier类的一个实例,

CommPortserialPort=portId.open("MyApp",60);

从而取得SerialPort.再请用它的addEventListener方法为它添加监听器.

serialPort.addEventListener(newMyPortListener());

SerialPort的监听罂必须继承于SerialPortEventListener接口。当有任何SerialPort的

事件发生时,将自动调用监听器中的serialEvent方法。SerialEvent有以下几种类型:

BI-通讯中断.

CD-载波检测.

CTS-清除发送.

DATA_AVAILABLE-有数据到达.

DSR-数据设备准备好.

FE-顿错误.

OE-溢位错误.

OUTPUT_BUFFER_EMPTY-输出缓冲区已清空.

PE-奇偶校验错.

RI-振铃指示.

下面是一个监听器的示例,

publicvoidMyPortListenerimplementsSerialPortEventListener

这个监听器只是简单打印每个发牛•的事件名称。而对于大多数应用程序来说,通常关心是

DATA_AVAILABLE事件,当数据从外部设备传送到端口上来时将触发此事件。此时就可以使

用前面提到过的方法,serialPort.getlnputStreamO^^Inputstream中读取数据了。

完整的程序

为节省篇幅,本文只提供了一些代码片断来都助读者来理解CommAPI的用法。你可以从

CommAPI的开发包中取得完整的可运行的演示程序。请先下载了commAPI的开发包,解

压之后有一个名为Sample的目录,里面有几个演示程序,分别是:

1)BlackBox:ASerialPortSlackBoxapplication.

2)ParallelBlackBox:AParallelPortBlackBoxapplication

3)ScrialDcmo:AsimplerScrialPortsampleapplication

4)Simple:Averysimplecommapplication

5)NullDriver:Atemplatefordriverwriters.Canbeusedasthestartingpoint

towriteadriverfortheCommAPI.

6)porting:AtemplateCommPortldentifierjavafileforpeopleinterested

inportingtheCommAPItoanewplatform.

其中.第1),3),4)是关于rs232通讯的演示程序.而其它的.2

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论