- 浏览: 34928 次
- 性别:
- 来自: 兴宁
最新评论
-
二农戏猪:
64位系统无法使用Error loading win32com ...
Java程序与RSR232串口通讯小练手 -
52mcu:
为啥用eclipse编译楼主的代码有如下错误,如何处理??// ...
Java程序与RSR232串口通讯小练手 -
一介布衣在江湖:
getResourceAsStream
Java程序与RSR232串口通讯小练手 -
mcw2:
win10 64 位 jdk1.8 可以吗?
Java程序与RSR232串口通讯小练手 -
15898whh:
怎么显示 未找到急求可用的串行端口号,程序无法启动! ...
Java程序与RSR232串口通讯小练手
一直以来都是在学习J2EE方面的应用系统开发,从未想过用JAVA来编写硬件交互程序,不过自己就是喜欢尝试一些未曾接触的新东西。在网上搜索了些资源,了解到JAVA写串口通讯的还是蛮多的,那么便着手准备开发调试环境。软件程序开发环境搭建不成问题,可这硬件环境就有点犯难啦。更何况自己用的是笔记本哪来的串口呀,再说要是真拿这串口硬件来自己也不会弄,随即想到了虚拟机,觉得这东西应该也有虚拟的吧,果真跟自己的猜测一样还真有这东西,顺便也下载了个串口小助手做为调试之用。下面就先看看软件环境的搭建:
1.下载comm.jar、win32com.dll和javax.comm.properties。 (附件提供下载)
介绍:comm.jar提供了通讯用的java API,win32com.dll提供了供comm.jar调用的本地驱动接口,javax.comm.properties是这个驱动的类配置文件
2.拷贝javacomm.jar到X:\jre\lib\ext目录下面;
3.拷贝javax.comm.properties到X:\jre\lib目录下面;
4.拷贝win32com.dll到X:\jre\bin目录下面;
5.更新下IDE里面的JDK环境,如下图:
接着是硬件虚拟环境安装虚拟串口,这里我用的是VSPD6.0(附件提供下载),安装好后启动VSPD添加我们所需要的端口,注意这里是按组的方式添加的,例如COM1和COM2是一组同时添加,以此类推。如下图所示:
Enumeration<?> en = CommPortIdentifier.getPortIdentifiers(); CommPortIdentifier portId; while (en.hasMoreElements()) { portId = (CommPortIdentifier) en.nextElement(); // 如果端口类型是串口,则打印出其端口信息 if (portId.getPortType() == CommPortIdentifier.PORT_SERIAL) { System.out.println(portId.getName()); } }
最后要解决的就是与串口数据交互的问题。在这个问题上,最主要的难点就是数据读取,因为我们不知道端口什么时候会有数据到来,也不知数据长度如何。通常,串口通信应用程序有两种模式,一种是实现SerialPortEventListener接口,监听各种串口事件并作相应处理;另一种就是建立一个独立的接收线程专门负责数据的接收。参考众多老前辈的代码后,下面就采用第一种方式写了个简单的助手程序,具体的实现请看详细代码,如下:
package com.elkan1788.view; import java.awt.BorderLayout; import java.awt.Button; import java.awt.Color; import java.awt.Font; import java.awt.GridLayout; import java.awt.Image; import java.awt.TextArea; import java.awt.TextField; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.util.ArrayList; import java.util.Enumeration; import java.util.List; import java.util.TooManyListenersException; import javax.comm.CommPortIdentifier; import javax.comm.NoSuchPortException; import javax.comm.PortInUseException; import javax.comm.SerialPort; import javax.comm.SerialPortEvent; import javax.comm.SerialPortEventListener; import javax.comm.UnsupportedCommOperationException; import javax.imageio.ImageIO; import javax.swing.JComboBox; import javax.swing.JFrame; import javax.swing.JLabel; import javax.swing.JOptionPane; import javax.swing.JPanel; import javax.swing.SwingConstants; import javax.swing.border.EmptyBorder; public class JavaRs232 extends JFrame implements ActionListener, SerialPortEventListener { /** * JDK Serial Version UID */ private static final long serialVersionUID = -7270865686330790103L; protected int WIN_WIDTH = 380; protected int WIN_HEIGHT = 300; private JComboBox<?> portCombox, rateCombox, dataCombox, stopCombox, parityCombox; private Button openPortBtn, closePortBtn, sendMsgBtn; private TextField sendTf; private TextArea readTa; private JLabel statusLb; private String portname, rate, data, stop, parity; protected CommPortIdentifier portId; protected Enumeration<?> ports; protected List<String> portList; protected SerialPort serialPort; protected OutputStream outputStream = null; protected InputStream inputStream = null; protected String mesg; protected int sendCount, reciveCount; /** * 默认构造函数 */ public JavaRs232() { super("Java RS-232串口通信测试程序 凡梦星尘"); setSize(WIN_WIDTH, WIN_HEIGHT); setLocationRelativeTo(null); Image icon = null; try { icon = ImageIO.read(JavaRs232.class.getResourceAsStream("/res/rs232.png")); } catch (IOException e) { showErrMesgbox(e.getMessage()); } setIconImage(icon); setResizable(false); scanPorts(); initComponents(); setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); setVisible(true); } /** * 初始化各UI组件 * @since 2012-3-22 下午11:56:39 */ public void initComponents() { // 共用常量 Font lbFont = new Font("微软雅黑", Font.TRUETYPE_FONT, 14); // 创建左边面板 JPanel northPane = new JPanel(); northPane.setLayout(new GridLayout(1, 1)); // 设置左边面板各组件 JPanel leftPane = new JPanel(); leftPane.setOpaque(false); leftPane.setLayout(new GridLayout(3,2)); JLabel portnameLb = new JLabel("串口号:"); portnameLb.setFont(lbFont); portnameLb.setHorizontalAlignment(SwingConstants.RIGHT); portCombox = new JComboBox<String>((String [])portList.toArray(new String[0])); portCombox.addActionListener(this); JLabel databitsLb = new JLabel("数据位:"); databitsLb.setFont(lbFont); databitsLb.setHorizontalAlignment(SwingConstants.RIGHT); dataCombox = new JComboBox<Integer>(new Integer[]{5, 6, 7, 8}); dataCombox.setSelectedIndex(3); dataCombox.addActionListener(this); JLabel parityLb = new JLabel("校验位:"); parityLb.setFont(lbFont); parityLb.setHorizontalAlignment(SwingConstants.RIGHT); parityCombox = new JComboBox<String>(new String[]{"NONE","ODD","EVEN","MARK","SPACE"}); parityCombox.addActionListener(this); // 添加组件至面板 leftPane.add(portnameLb); leftPane.add(portCombox); leftPane.add(databitsLb); leftPane.add(dataCombox); leftPane.add(parityLb); leftPane.add(parityCombox); //创建右边面板 JPanel rightPane = new JPanel(); rightPane.setLayout(new GridLayout(3,2)); // 设置右边面板各组件 JLabel baudrateLb = new JLabel("波特率:"); baudrateLb.setFont(lbFont); baudrateLb.setHorizontalAlignment(SwingConstants.RIGHT); rateCombox = new JComboBox<Integer>(new Integer[]{2400,4800,9600,14400,19200,38400,56000}); rateCombox.setSelectedIndex(2); rateCombox.addActionListener(this); JLabel stopbitsLb = new JLabel("停止位:"); stopbitsLb.setFont(lbFont); stopbitsLb.setHorizontalAlignment(SwingConstants.RIGHT); stopCombox = new JComboBox<String>(new String[]{"1","2","1.5"}); stopCombox.addActionListener(this); openPortBtn = new Button("打开端口"); openPortBtn.addActionListener(this); closePortBtn = new Button("关闭端口"); closePortBtn.addActionListener(this); // 添加组件至面板 rightPane.add(baudrateLb); rightPane.add(rateCombox); rightPane.add(stopbitsLb); rightPane.add(stopCombox); rightPane.add(openPortBtn); rightPane.add(closePortBtn); // 将左右面板组合添加到北边的面板 northPane.add(leftPane); northPane.add(rightPane); // 创建中间面板 JPanel centerPane = new JPanel(); // 设置中间面板各组件 sendTf = new TextField(42); readTa = new TextArea(8,50); readTa.setEditable(false); readTa.setBackground(new Color(225,242,250)); centerPane.add(sendTf); sendMsgBtn = new Button(" 发送 "); sendMsgBtn.addActionListener(this); // 添加组件至面板 centerPane.add(sendTf); centerPane.add(sendMsgBtn); centerPane.add(readTa); // 设置南边组件 statusLb = new JLabel(); statusLb.setText(initStatus()); statusLb.setOpaque(true); // 获取主窗体的容器,并将以上三面板以北、中、南的布局整合 JPanel contentPane = (JPanel)getContentPane(); contentPane.setLayout(new BorderLayout()); contentPane.setBorder(new EmptyBorder(0, 0, 0, 0)); contentPane.setOpaque(false); contentPane.add(northPane, BorderLayout.NORTH); contentPane.add(centerPane, BorderLayout.CENTER); contentPane.add(statusLb, BorderLayout.SOUTH); } /** * 初始化状态标签显示文本 * @return String * @since 2012-3-23 上午12:01:53 */ public String initStatus() { portname = portCombox.getSelectedItem().toString(); rate = rateCombox.getSelectedItem().toString(); data = dataCombox.getSelectedItem().toString(); stop = stopCombox.getSelectedItem().toString(); parity = parityCombox.getSelectedItem().toString(); StringBuffer str = new StringBuffer("当前串口号:"); str.append(portname).append(" 波特率:"); str.append(rate).append(" 数据位:"); str.append(data).append(" 停止位:"); str.append(stop).append(" 校验位:"); str.append(parity); return str.toString(); } /** * 扫描本机的所有COM端口 * @since 2012-3-23 上午12:02:42 */ public void scanPorts() { portList = new ArrayList<String>(); Enumeration<?> en = CommPortIdentifier.getPortIdentifiers(); CommPortIdentifier portId; while(en.hasMoreElements()){ portId = (CommPortIdentifier) en.nextElement(); if(portId.getPortType() == CommPortIdentifier.PORT_SERIAL){ String name = portId.getName(); if(!portList.contains(name)) { portList.add(name); } } } if(null == portList || portList.isEmpty()) { showErrMesgbox("未找到可用的串行端口号,程序无法启动!"); System.exit(0); } } /** * 打开串行端口 * @since 2012-3-23 上午12:03:07 */ public void openSerialPort() { // 获取要打开的端口 try { portId = CommPortIdentifier.getPortIdentifier(portname); } catch (NoSuchPortException e) { showErrMesgbox("抱歉,没有找到"+portname+"串行端口号!"); setComponentsEnabled(true); return ; } // 打开端口 try { serialPort = (SerialPort) portId.open("JavaRs232", 2000); statusLb.setText(portname+"串口已经打开!"); } catch (PortInUseException e) { showErrMesgbox(portname+"端口已被占用,请检查!"); setComponentsEnabled(true); return ; } // 设置端口参数 try { int rate = Integer.parseInt(this.rate); int data = Integer.parseInt(this.data); int stop = stopCombox.getSelectedIndex()+1; int parity = parityCombox.getSelectedIndex(); serialPort.setSerialPortParams(rate,data,stop,parity); } catch (UnsupportedCommOperationException e) { showErrMesgbox(e.getMessage()); } // 打开端口的IO流管道 try { outputStream = serialPort.getOutputStream(); inputStream = serialPort.getInputStream(); } catch (IOException e) { showErrMesgbox(e.getMessage()); } // 给端口添加监听器 try { serialPort.addEventListener(this); } catch (TooManyListenersException e) { showErrMesgbox(e.getMessage()); } serialPort.notifyOnDataAvailable(true); } /** * 给串行端口发送数据 * @since 2012-3-23 上午12:05:00 */ public void sendDataToSeriaPort() { try { sendCount++; outputStream.write(mesg.getBytes()); outputStream.flush(); } catch (IOException e) { showErrMesgbox(e.getMessage()); } statusLb.setText(" 发送: "+sendCount+" 接收: "+reciveCount); } /** * 关闭串行端口 * @since 2012-3-23 上午12:05:28 */ public void closeSerialPort() { try { if(outputStream != null) outputStream.close(); if(serialPort != null) serialPort.close(); serialPort = null; statusLb.setText(portname+"串口已经关闭!"); sendCount = 0; reciveCount = 0; sendTf.setText(""); readTa.setText(""); } catch (Exception e) { showErrMesgbox(e.getMessage()); } } /** * 显示错误或警告信息 * @param msg 信息 * @since 2012-3-23 上午12:05:47 */ public void showErrMesgbox(String msg) { JOptionPane.showMessageDialog(this, msg); } /** * 各组件行为事件监听 */ public void actionPerformed(ActionEvent e) { if(e.getSource() == portCombox || e.getSource() == rateCombox || e.getSource() == dataCombox || e.getSource() == stopCombox || e.getSource() == parityCombox){ statusLb.setText(initStatus()); } if(e.getSource() == openPortBtn){ setComponentsEnabled(false); openSerialPort(); } if(e.getSource() == closePortBtn){ if(serialPort != null){ closeSerialPort(); } setComponentsEnabled(true); } if(e.getSource() == sendMsgBtn){ if(serialPort == null){ showErrMesgbox("请先打开串行端口!"); return ; } mesg = sendTf.getText(); if(null == mesg || mesg.isEmpty()){ showErrMesgbox("请输入你要发送的内容!"); return ; } sendDataToSeriaPort(); } } /** * 端口事件监听 */ public void serialEvent(SerialPortEvent event) { switch (event.getEventType()) { case SerialPortEvent.BI: case SerialPortEvent.OE: case SerialPortEvent.FE: case SerialPortEvent.PE: case SerialPortEvent.CD: case SerialPortEvent.CTS: case SerialPortEvent.DSR: case SerialPortEvent.RI: case SerialPortEvent.OUTPUT_BUFFER_EMPTY: break; case SerialPortEvent.DATA_AVAILABLE: byte[] readBuffer = new byte[50]; try { while (inputStream.available() > 0) { inputStream.read(readBuffer); } StringBuilder receivedMsg = new StringBuilder("/-- "); receivedMsg.append(new String(readBuffer).trim()).append(" --/\n"); readTa.append(receivedMsg.toString()); reciveCount++; statusLb.setText(" 发送: "+sendCount+" 接收: "+reciveCount); } catch (IOException e) { showErrMesgbox(e.getMessage()); } } } /** * 设置各组件的开关状态 * @param enabled 状态 * @since 2012-3-23 上午12:06:24 */ public void setComponentsEnabled(boolean enabled) { openPortBtn.setEnabled(enabled); openPortBtn.setEnabled(enabled); portCombox.setEnabled(enabled); rateCombox.setEnabled(enabled); dataCombox.setEnabled(enabled); stopCombox.setEnabled(enabled); parityCombox.setEnabled(enabled); } /** * 运行主函数 * @param args * @since 2012-3-23 上午12:06:45 */ public static void main(String[] args) { new JavaRs232(); } }
2.端口检测
3. 通讯测试
最后再抽空来美化程序下,效果更漂亮,谁还会说JAVA程序的界面丑陋呢,呵呵...
第一次发文虽没有什么技术含量但也实属不易哪,欢迎大家拍砖,嘻嘻....
- JavaRS232小练手.zip (4.3 MB)
- 下载次数: 1404
评论
Error loading win32com: java.lang.UnsatisfiedLinkError: D:\Java\jdk1.7.0_79\jre\bin\win32com.dll: Can't load IA 32-bit .dll on a AMD 64-bit platform
为啥用eclipse编译楼主的代码有如下错误,如何处理??
//以下语句被画上红色波浪线
import javax.comm.CommPortIdentifier;
import javax.comm.NoSuchPortException;
import javax.comm.PortInUseException;
import javax.comm.SerialPort;
import javax.comm.SerialPortEvent;
import javax.comm.SerialPortEventListener;
import javax.comm.UnsupportedCommOperationException;
提示错误
Access restriction: The type UnsupportedCommOperationException is not accessible due to restriction on required
library C:\Program Files (x86)\Java\jre7\lib\ext\comm.jar
运行代码时的错误
Exception in thread "main" java.lang.IllegalArgumentException: input == null!
at javax.imageio.ImageIO.read(Unknown Source)
at com.elkan1788.view.JavaRs232.<init>(JavaRs232.java:85)
at com.elkan1788.view.JavaRs232.main(JavaRs232.java:430)
相关推荐
在Java中实现串口通信,通常会用到javax.comm库,这是一个为Java应用程序提供串口通信功能的API。然而,这个库在Java 6之后就没有得到官方支持,所以在现代的Java开发中,开发者更倾向于使用第三方库,如RXTX或...
使用网页读取串口电子秤的重量,使用插件方式&JS方式实现, 在网页的特定地方显示当前通过RS232链接的电子秤上的重量数据。...我的博客 java RS232串口通讯上有对此有相关方法,希望能对大家有帮助。
**RS232串口通讯程序** 在计算机技术领域,串行通信是一种常见的数据传输方式,其中RS232标准是应用最为广泛的串行接口之一。这个标准由电子工业协会(EIA)制定,用于规范设备间的物理连接和电气特性。在本案例中...
VB与RS232串口通讯,可以用来学习控制单片机
以下是对这个"android RS232串口读写程序"的详细解析: 1. **RS232串口通信**: RS232是最早的标准串行接口之一,用于设备之间的通信。它定义了数据位、停止位、校验位、波特率等通信参数,并通过DB9或DB25连接器...
RS232串口通讯是计算机硬件接口技术中的一种,广泛应用于设备间的串行数据传输。这个通信协议标准,即EIA/TIA-232(Electrical Industry Association/Telecommunications Industry Association 232),定义了数据...
综上所述,"RS232串口通讯DLL源代码"涉及的关键技术包括RS232串口通信协议,DLL的使用与开发,以及在VC++6.0环境下进行编译和调试。通过理解这些知识点,开发者可以有效地创建自己的串口通信功能库,为各种应用程序...
RS232串口驱动程序是计算机硬件与外部设备间通信的一种关键技术,广泛应用于工业控制、数据通信和设备调试等领域。RS232,全称为EIA/TIA-232标准,是由电子工业协会(Electronic Industries Alliance, EIA)和电信...
在实际应用中,`RS232串口通讯CSharp`示例可能包含了一个或多个C#源代码文件,展示了如何创建、配置、打开串口,以及如何进行数据交换。通过学习和理解这些示例,开发者可以更好地掌握在C#中使用RS232进行串口通信的...
总结,VB RS232串口通讯源程序是实现设备间简单而有效的数据交换手段,虽然现代技术有更快的通信协议,但RS232因其兼容性与稳定性,在许多场合仍占据一席之地。掌握VB中的串口通信技术,对于开发者来说是一项必备...
1、需要一根Rs232通讯线;2、2台PC机; 之前用c#实现的一个Rs232通讯,最近工厂新增产线,又翻出来看了看。 实现了两台pc机通过rs232通讯线来进行简单的发送、接收数据, 代码简单通俗易懂,适合入门。
而“cordova集成RS232串口通讯”这个主题,涉及到的是如何在Cordova应用中实现与硬件设备通过RS232串行接口进行通信。 首先,了解RS232串口通讯的基础知识是必要的。RS232,全称是“Recommended Standard 232”,是...
这个"delphi RS232 计算机串口通讯源程序"正是为开发者提供了利用Delphi进行串口通信的实例代码。 RS232,全称是EIA/TIA-232标准,是一种早期定义的串行物理接口标准,用于实现计算机和其他设备之间的通信。它的...
总结来说,JAVA rs232 串口通信技术是Java编程与硬件设备交互的关键工具,通过`w32comm.dll`、`comm.jar`等组件,开发者可以构建跨平台的串口通信应用。正确理解和使用这些工具,可以极大地提升硬件设备与软件应用的...
[013]RS232串口通讯工具源码,VC++程序,方便监控,自动化等行业的应用.zip上位机开发VC串口学习资料源码下载[013]RS232串口通讯工具源码,VC++程序,方便监控,自动化等行业的应用.zip上位机开发VC串口学习资料源码...
串口通讯程序
**RS232串口通讯**是计算机通信中常见的一种技术,主要用于设备间的短距离通信。在本实例中,我们关注的是如何使用VB.NET进行RS232串口的编程,包括串口参数的设置、数据接收以及数据发送。下面将详细解释这些知识点...
本教程将详细讲解如何使用C++编程语言和MFC(Microsoft Foundation Classes)库来实现RS232串口通讯。 RS232,全称为“推荐标准232”,是定义计算机之间串行通信接口的一种标准。它允许数据以串行的方式传输,通常...
标题中的“vb RS232串口通讯程序,源码附图教程”指的是一个使用Visual Basic(VB)编写的程序,该程序实现了通过RS232串行接口进行通信的功能。RS232是一种标准的串行物理接口,广泛用于设备间的通信,如计算机与...