您好,欢迎访问三七文档
小技巧:如何辨别TXD和RXD端口?搞电子的人手边应该常备一个电表,用来测测电压,电阻什么的会很有用。你只要分别测一下RS232端口的2--5或3--5针脚之间的电压,通常TXD针脚与GND之间会有3~15V左右的负电压,表示它是TXD针脚。安装JavaCommunicationsAPISun的J2SE中并没有直接提供以上提到的任何一种串行通讯协议的开发包,而是以独立的jar包形式发布在java.sun.com网站上(从这里下载)----即comm.jar,称之为JavatmCommunicationsAPI,它是J2SE的标准扩展。comm.jar并不是最近才有,早在1998年时,sun就已经发布了这个开发包。comm.jar分别提供了对常用的RS232串行端口和IEEE1284并行端口通讯的支持。目前sun发布的comm.jar只有Windows和Solaris平台两个版本,如果你需要Linux平台下的,可以在~kevinh/linuxcomm.html找到。在使用comm.jar之前,必须知道如何安装它。这也是困扰许多初学javaRS232通讯者的一个难题。如果我们电脑上安装了JDK,它将同时为我们安装一份JRE(JavaRuntimeEntironment),通常我们运行程序时都是以JRE来运行的。所以以下的安装适用于JRE。如果你是用JDK来运行程序的,请将相应的改成。下载了comm.jar开发包后,与之一起的还有两个重要的文件,win32com.dll和javax.comm.properties。comm.jar提供了通讯用的javaAPI,而win32com.dll提供了供comm.jar调用的本地驱动接口。而javax.comm.properties是这个驱动的类配置文件。首先将comm.jar复制到\lib\ext目录。再将win21com.dll复制到你的RS232应用程序运行的目录,即user.dir。然后将javax.comm.properties复制到\lib目录。通讯前的准备如果你手头上没有现成的提供了标准RS232串口的设备,你可以将自己的电脑模拟成两台不同的串口设备。通常电脑主机后面的面板提供了两个9针的串口,请将这两个串口的2,3,5脚按前面介绍的方法连接。电子市场都有现成的连接头卖,请不要买那种封装的严严实实的接头,而要买用螺丝封装可以拆开的连接头,这样可以方便自己根据需要连接各个针脚。CommAPI基础我无意于在此详细描述CommAPI每个类和接口的用法,但我会介绍CommAPI的类结构和几个重要的API用法。所有的commAPI位于javax.comm包下面。从CommAPI的javadoc来看,它介绍给我们的只有区区以下13个类或接口:javax.comm.CommDriverjavax.comm.CommPortjavax.comm.ParallelPortjavax.comm.SerialPortjavax.comm.CommPortIdentifierjavax.comm.CommPortOwnershipListenerjavax.comm.ParallelPortEventjavax.comm.SerialPortEventjavax.comm.ParallelPortEventListener(extendsjava.util.EventListener)javax.comm.SerialPortEventListener(extendsjava.util.EventListener)javax.comm.NoSuchPortExceptionjavax.comm.PortInUseExceptionjavax.comm.UnsupportedCommOperationException下面讲解一下几个主要类或接口。1.枚举出系统所有的RS232端口在开始使用RS232端口通讯之前,我们想知道系统有哪些端口是可用的,以下代码列出系统中所有可用的RS232端口:Enumerationen=CommPortIdentifier.getPortIdentifiers();CommPortIdentifierportId;while(en.hasMoreElements()){portId=(CommPortIdentifier)en.nextElement();/*如果端口类型是串口,则打印出其端口信息*/if(portId.getPortType()==CommPortIdentifier.PORT_SERIAL){System.out.println(portId.getName());}}在我的电脑上以上程序输出以下结果:COM1COM2CommPortIdentifier类的getPortIdentifiers方法可以找到系统所有的串口,每个串口对应一个CommPortIdentifier类的实例。2.打开端口如果你使用端口,必须先打开它。try{CommPortserialPort=portId.open(MyApp,60);/*从端口中读取数据*/InputStreaminput=serialPort.getInputStream();input.read(...);/*往端口中写数据*/OutputStreamoutput=serialPort.getOutputStream();output.write(...)...}catch(PortInUseExceptionex){...}通过CommPortIdentifier的open方法可以返回一个CommPort对象。open方法有两个参数,第一个是String,通常设置为你的应用程序的名字。第二个参数是时间,即开启端口超时的毫秒数。当端口被另外的应用程序占用时,将抛出PortInUseException异常。在这里CommPortIdentifier类和CommPort类有什么区别呢?其实它们两者是一一对应的关系。CommPortIdentifier主要负责端口的初始化和开启,以及管理它们的占有权。而CommPort则是跟实际的输入和输出功能有关的。通过CommPort的getInputStream()可以取得端口的输入流,它是java.io.InputStream接口的一个实例。我们可以用标准的InputStream的操作接口来读取流中的数据,就像通过FileInputSteam读取文件的内容一样。相应的,CommPort的getOutputStream可以获得端口的输出流,这样就可以往串口输出数据了。3.关闭端口使用完的端口,必须记得将其关闭,这样可以让其它的程序有机会使用它,不然其它程序使用该端口时可能会抛出端口正在使用中的错误。很奇怪的是,CommPortIdentifier类只提供了开启端口的方法,而要关闭端口,则要调用CommPort类的close()方法。CommPort的输入流的读取方式与文件的输入流有些不一样,那就是你可能永远不知这个InputStream何时结束,除非对方的OutputStream向你发送了一个特定数据表示发送结束,你收到这个特定字符后,再行关闭你的InputStream。而comm.jar提供了两种灵活的方式让你读取数据。1.轮询方式(Polling)举个例子,你同GF相约一起出门去看电影,但你的GF好打扮,这一打扮可能就是半小时甚至一小时以上。这时你就耐不住了,每两分钟就催问一次“好了没?”,如此这样,直到你的GF说OK了才算完。这个就叫轮询(Polling)。在程序中,轮询通常设计成一个封闭的循环,当满足某个条件时即结束循环。刚才那个例子中,你的GF说“OK了!”,这个就是结束你轮询的条件。在单线程的程序中,当循环一直执行某项任务而又无法预知它何时结束时,此时你的程序看起来可能就像死机一样。在VB程序中,这个问题可以用在循环结构中插入一个doEvent语句来解决。而Java中,最好的方式是使用线程,就像以下代码片断一样。publicTestPortextendThread{...InputStreaminput=serialPort.getInputStream();StringBufferbuf=newStringBuffer();booleanstopped=false;...publicvoidrun(){try{while(!stopped)intch=input.read();if(ch==&aposq&apos||ch==&aposQ&apos){/*结束读取,关闭端口...*/stopped=true;...}else{buf.append((char)ch);...}}catch(InterruptedExceptione){}}}2.监听方式(listening)CommAPI支持标准的JavaBean型的事件模型。也就是说,你可以使用类似AddXXXListener这样的方法为一个串口注册自己的监听器,以监听方式进行数据读取。如要对端口监听,你必须先取得CommPortIdentifier类的一个实例,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{publicvoidserialEvent(SerialPortEventevt){switch(evt.getEventType()){caseSerialPortEvent.CTS:System.out.println(CTSeventoccured.);break;caseSerialPortEvent.CD:System.out.println(CDeventoccured.);break;caseSerialPortEvent.BI:System.out.println(BIeventoccured.);break;caseSerialPortEvent.DSR:System.out.println(DSReventoccured.);break;caseSerialPortEvent.FE:System.out.println(FEeventoccured.);break;caseSerialPortEvent.OE:System.out.println(OEeventoccured.);break;caseSerialPortEvent.PE:System.out.println(PEeventoccured.);break;caseSerialPortEvent.RI:System.out.println(RIeventoccured.);break;caseSerialPortEvent.OUTPUT_BUFFER_EMPTY:System.out.println(OUTPUT_BUFFER_EMPTYeventoccured.);break;caseSerialPortEvent.DATA_AVAILABLE:S
本文标题:java串口通信
链接地址:https://www.777doc.com/doc-6148777 .html