測试环境:

SDKOracle JRockit for Java version 6, Java Communication for Windows 2.0

OSWINDOWS7

外设:串口条形码扫描枪

ServerTomcat6

看了网上良莠不齐的关于Applet訪问串口的文章,总结起来所关注的问题无外乎下面3个:

1. 三个文件(comm.jarjavax.comm.propertieswin32com.dll)究竟应该存放在什么文件夹中?

2. 怎样实现代码?

3. Applet究竟应该这么部署?

一.关于第一个问题,网上大致是这样写的:

a) javax.comm.properties文件放在$JAVA_HOME/lib文件夹中;

b) win32comm.dll文件放在$JAVA_HOME/bin文件夹中;

c) comm.jar文件放在$JAVA_HOME/lib/ext文件夹中;

先不去讨论这些文件应不应该放在这些文件夹中,单从可行性方面讨论就不太符合WEB应用程序的做法。首先您不可能预知有多少client的存在,就算您预先知道也不可能在每一个client计算机上部署上述3个文件。好,您说能够提供使用手冊引导用户下载文件并依照手冊将上述文件部署到指定文件夹。可是您添加了用户的使用学习成本,用户不是IT专家,将本来应该由开发者完毕的任务转嫁给用户是否合适呢?

要解决问题,关键是要測试,測试Applet在执行时是这么载入这些文件的。经过重复的測试,最终搞清楚当中的来龙去脉:

1. javax.comm.properties文件能够丢弃,由于通过编程的方法能够在Applet中动态载入串口驱动程序,所以该文件存不存在无所谓。

2. comm.jar文件是基本的串口訪问类库,能够通过在Applet执行时载入(通过ARCHIVE參数指定,后面有具体的样例),所以也没有必要事先部署到client计算机上。

3. 最关键的是win32comm.dll文件,该文件是用C写的串(并)口驱动程序,Java通过JNI调该文件里的函数来实现对串(并)口的訪问。所以此文件不可或缺。要将该文件部署到client仅仅能通过下载的方式实现,即在Applet执行时检查指定文件夹中是否存在win32comm.dll文件,假设不存在则将server端的win32comm.dll文件下载到client的指定文件夹中,最后再动态装载驱动程序。关于win32comm.dll文件究竟要部署到什么文件夹中,经过測试发现该文件仅仅要存在于由java.library.path系统变量指定的任一文件夹中就可以,该系统变量能够通过System.getProperty(“java.library.path”)方法获取。下面是在我的机器中使用该方法获取的值:

C:/Program Files/Java/jrmc-3.1.2-1.6.0/bin;.;C:/Windows/system32;C:/Windows;C:/Program Files/Java/jrockit-R27.6.5-jre1.6.0_14/bin;C:/Windows/system32;C:/Windows;C:/Windows/System32/Wbem;C:/Windows/System32/WindowsPowerShell/v1.0/;C:/Program Files/ATI Technologies/ATI.ACE/Core-Static;C:/Program Files/Toshiba/Bluetooth Toshiba Stack/sys/;C:/Program Files/SecureCRT/;C:/Program Files/MySQL/MySQL Server 5.1/bin;C:/Tomcat6.0/bin;E:/software/java/jdk/ibm_sdk60/bin;C:/MinGW/bin;C:/Program Files/Java/jrmc-3.1.2-1.6.0/bin;

通过上述分析,我们已经理清了三个文件存放位置,接下来就是怎样详细的实现代码了。

二.代码实现

a) 下载win32comm.dll文件到client:

先看代码的实现:

private static final String LIB_PATH_SUFFIX = "system32";

private static final String DLL_FILE = "win32com.dll";

try {

// 获取载入库时搜索的路径列表

String dirs = System.getProperty("java.library.path");

String[] libs = dirs.split(";");

String libPath = "";

for (String lib : libs) {

if (lib.toLowerCase().endsWith(LIB_PATH_SUFFIX)) {

libPath = lib;

break;

}

}

File dll = new File(libPath, DLL_FILE);

if (!dll.exists()) {

URL url = new URL(super.getCodeBase() + DLL_FILE);

InputStream is = url.openConnection().getInputStream();

FileOutputStream fos = new FileOutputStream(dll);

byte[] buf = new byte[256]; // 读取缓存

int len = 0;

while ((len = is.read(buf)) != -1) {

fos.write(buf, 0, len);

}

fos.flush();

fos.close();

is.close();

System.out.println("创建文件完毕[" + dll + "].");

}

} catch (MalformedURLException e) {

e.printStackTrace();

} catch (IOException e) {

e.printStackTrace();

}

这段代码的主要算法例如以下:

1. 通过System.getProperty(“java.library.path”)方法获取载入库时搜索的文件夹字符串,每一个文件夹之间是用分号隔开的;

2. 将文件夹字符串按分号拆分成字符串数组,然后取当中的任一一个就可以。只是我喜欢取“C:/Windows/system32文件夹;

3. 然后实例化一个File对象,该对象存有指向C:/Windows/system32/win32comm.dll文件的句柄。检測该文件是否存在,假设存在则不做不论什么处理,否则进行下载处理;

4. 假设须要下载文件,则先通过getCodeBase()方法获取server端的根URL对象。然后构造一个指向server端win32comm.dll文件的URL对象;

5. 通过URLopenConnection().getInputStream()方法获取InputStream流准备读取;

6. 在client实例化一个FileOutputStream对象,并将输出流写入到client的文件里(C:/Windows/system32/win32comm.dll),完毕文件的下载。事实上文件的下载是通过http协议完毕的,即httpclient。所以server端须要TOMCAT或其它WEBserver软件。

b) 读取条形码

主要实现Applet3个方法:initstartdestroy方法。先看init方法:

private String driverName = "com.sun.comm.Win32Driver";

public void init() {

try {

System.loadLibrary("win32com");

CommDriver driver = (CommDriver)Class.forName(driverName).newInstance();

driver.initialize();

} catch (Exception e) {

System.err.println(e);

}

}

init方法在Applet载入时运行一次且仅一次。所以init方法中适合装载和初始化驱动程序,即载入win32comm.dll文件。

条码扫描设备与调制解调器不同,调制解调器使用“密步”的通信方式,即请求-应答模式。而条码扫描设备是事件驱动的,仅仅有在扫描了条码之后,才干读取串口的数据,所以使用请求-应答模式肯定不行。为了解决问题,Applet必须实现SerialPortEventListener接口,以便在有数据到达时运行特定的方法。另外还须要启动一个线程来等待数据的到达。所以Applet类的签名例如以下:

public class SerialPortApplet extends JApplet implements Runnable,

SerialPortEventListener {

public void run() {

}

public void serialEvent(SerialPortEvent event) {

}

}

当中serialEvent(SerialPortEvent event)方法就是须要实现的接口方法,当串口有数据到达时,则会运行该方法中的代码。

start方法在init方法运行完成之后运行,在Applet的整个生命周期中,start方法能够被运行多次。所以start方法能够用来实现寻找可用port,打开port,设置port參数,等待数据到达以及数据处理等代码,代码例如以下所看到的:

private CommPortIdentifier portId;

private StringBuilder barcode = new StringBuilder();

private InputStream is;

private boolean over = false; //退出线程的标志

private SerialPort serialPort;

static {

System.setSecurityManager(null); //禁用安全管理器(必须写)

}

public void start() {

Enumeration ports = CommPortIdentifier.getPortIdentifiers();

while (ports.hasMoreElements()) {

portId = (CommPortIdentifier) ports.nextElement();

if (portId.getPortType() == CommPortIdentifier.PORT_SERIAL) { // 是串口

if (portId.getName().equals("COM1")) {

break;

}

}

}

try {

serialPort = (SerialPort) portId.open("App1",2000); // 打开port

is = serialPort.getInputStream();

serialPort.addEventListener(this); // 注冊监听器

serialPort.notifyOnDataAvailable(true); // 数据达到时发出通知

serialPort.setSerialPortParams(9600, SerialPort.DATABITS_8,SerialPort.STOPBITS_1, SerialPort.PARITY_NONE); // 设置port參数

} catch (PortInUseException e) {

System.err.println(e);

} catch (IOException e) {

System.err.println(e);

} catch (TooManyListenersException e) {

System.err.println(e);

} catch (UnsupportedCommOperationException e) {

System.err.println(e);

}

// 启动线程

Thread t = new Thread(this);

t.start();

try {

t.join();// 等待线程结束

} catch (InterruptedException e) {

}

System.out.println("barcode[" + barcode + "]");

}

线程接口的实现和监听器接口的实现代码例如以下所看到的:

public void run() {

while (!over) {

}

try {

if (is != null) {

is.close();

}

if (serialPort != null) {

serialPort.close();

}

} catch (IOException e) {

System.out.println(e);

}

}

public void serialEvent(SerialPortEvent event) {

switch (event.getEventType()) {

case SerialPortEvent.DATA_AVAILABLE: //数据到达时运行

try {

while (true) {

int b = is.read(); // 假设读取不到数据则会堵塞

if (b == 10 || b == 13) { // 假设读到回车或换行则表示读取完毕

over = true;

break;

} else {

barcode.append(new String(new byte[] { (byte) b }));

}

}

} catch (IOException e) {

System.err.println(e);

}

}

}

destroy方法在Applet销毁时运行且运行一次,所以能够该方法中编写资源释放的代码,代码实现例如以下:

public void destroy() {

try {

if (is != null) {

is.close();

}

if (serialPort != null) {

serialPort.close();

}

} catch (IOException e) {

System.out.println(e);

}

}

c) HTML页面(index.html)

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">

<html xmlns="http://www.w3.org/1999/xhtml">

<head>

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" />

<title></title>

</head>

<body>

<!--"CONVERTED_APPLET"-->

<!-- HTML CONVERTER -->

<OBJECT classid="clsid:8AD9C840-044E-11D1-B3E9-00805F499D93"

WIDTH = "250" HEIGHT = "140"  codebase="http://java.sun.com/update/1.6.0/jinstall-6u17-windows-i586.cab#Version=6,0,0,4">

<PARAM NAME = CODE VALUE = "org.oakman.applets.SerialPortApplet.class" >

<PARAM NAME = CODEBASE VALUE = "." >

<PARAM NAME = ARCHIVE VALUE = "comm.jar, serial_port.jar" >

<PARAM NAME="type" VALUE="application/x-java-applet;version=1.6">

<PARAM NAME="scriptable" VALUE="false">

<COMMENT>

<EMBED type="application/x-java-applet;version=1.6"  CODE = "org.oakman.applets.SerialPortApplet.class" CODEBASE = "." ARCHIVE = " comm.jar,serial_port.jar" WIDTH = "250" HEIGHT = "140"  scriptable=false pluginspage="http://java.sun.com/products/plugin/index.html#download"><NOEMBED>

</NOEMBED>

</EMBED>

</COMMENT>

</OBJECT>

<!--"END_CONVERTED_APPLET"-->

</body>

</html>

当中比較重要的是ARCHIVE參数,能够将Applet须要用到的全部jar类库都在这个參数中进行设置,多个jar文件之间用逗号进行分隔。假设client没有安装JRE,则首先会要求用户下载JRE,正常情况下,下载JRE须要10分钟左右的时间。

经过上述步骤,我们完毕了全部须要实现的代码。将全部Java代码打包成jar文件,然后和html文件一起部署到TOMCAT6中,启动,一切正常,然后很高兴的打开浏览器,输入url,双眼充满期望的等待令人兴奋的一幕。只是……,报错了!很的沮丧!错误提示没有訪问权限!!郁闷中……

三.部署

Java号称是最安全的,所以Applet作为在网络上能够随意传播的小应用程序当然更加须要安全。所以在默认情况下Applet仅仅能执行在JVM的沙箱中。不能訪问client的不论什么资源,包含文件系统的读写,网络套接字等。所以出现上述错误理所当然,反而证明了Java的确很的安全。

为了走出沙箱,我们必须对Applet进行签名:

1. 创建密钥:

使用例如以下命令进行密钥的创建,这里使用RSA算法而不是Java默认的DSA算法

keytool -genkey -alias oakman -keyalg RSA

执行时会要求您输入密钥库的口令,并要求您输入名字,组织和位置等信息。填好后就会产生密钥(密钥文件在用户文件夹中,找.keystore文件)。

2. CA订购签名证书:

为了从CA订购签名证书,你须要从密钥库中导出证书签名申请(CSR文件)。使用例如以下命令

keytool -certreq -alias oakman -file oakman.csr

该命令会在当前文件夹中产生一个oakman.csr文件,然后你能够用这个文件以及证明你身份的证明或证件以及数KRMBCA那里申请你的证书(比較常见的CAVerisign)。假设申请成功,CA会给你一个BASE64编码证书,你就能够把它导入到密钥库中给自己编写的Applet进行签名了,导入命令例如以下:

keytool -import -alias oakman -file oakman.cer

当中oakman.cer就是CA给你的证书。

3. Appletjar文件进行签名:

已经将CA的证书导入到密钥库中,那么就能够对自己编写的Applet进行签名了(当然,先要打包成JAR包),使用例如以下命令:

jarsigner serial_port.jar oakman

jarsigner comm.jar oakman

Applet全部用到的jar文件都要进行签名。

经过上述步骤,我们的Applet就能够走出沙箱了,能够訪问client的不论什么资源,包含文件系统,外设以及网络套接字等。将条形码扫码枪接上我笔记本的串口,找了一本书,对着书上的条码一扫,哔!!扫描成功,接着在Java控制台出现一串数字,OK!最终搞定。(注:因为我笔记本没有串口,所以通过USB转串口来模拟出串口,可能是因为驱动程序的原因,假设在没有关闭串口的情况关闭IE浏览器,则操作系统必定死机,仅仅能手动重新启动操作系统。)

当然,作为測试,我们没有必要花数KRMBCA那里申请证书,所以能够将步骤2省略。在生成密钥库之后直接对JAR文件进行签名。

四.部署文件夹

下面为我机器上TOMCAT中应用程序的部署文件夹:

pay

|-- WEB-INF

|-- web.xml

|-- comm.jar

|-- serial_port.jar

|-- index.html

转载于:https://www.cnblogs.com/mengfanrong/p/4006098.html

Java Applet读写client串口——终极篇相关推荐

  1. Java文件读写操作(c站最全!一篇就够了)

    写在前面   你们好,我是小庄.很高兴能和你们一起学习Java.如果您对Java感兴趣的话可关注我的动态.   写博文是一种习惯,在这过程中能够梳理和巩固知识. Java文件读写操作 一.File类 ...

  2. 115个Java面试题和答案——终极列表(下)

    转载自   115个Java面试题和答案--终极列表(下) 第一篇讨论了面向对象编程和它的特点,关于Java和它的功能的常见问题,Java的集合类,垃圾收集器,本章主要讨论异常处理,Java小应用程序 ...

  3. 115 个 Java 面试题和答案——终极重点(下)

    题目:115 个 Java 面试题和答案--终极(下) 第一篇讨论了面向对象编程和它的特点,关于 Java 和它的功能的常见问题,Java 的集合类,垃圾收集器,本章主要讨论异常处理,Java 小应用 ...

  4. HIDL示例-JAVA服务创建-Client验证-Android10.0 HwBinder通信原理(四)

    摘要:本节主要来讲解Android10.0 JAVA层的HIDL服务创建和JAVA层的Client验证 阅读本文大约需要花费15分钟. 文章首发微信公众号:IngresGe 专注于Android系统级 ...

  5. 115个Java面试题和答案——终极列表(上)

    转载自  115个Java面试题和答案--终极列表(上) 本文我们将要讨论Java面试中的各种不同类型的面试题,它们可以让雇主测试应聘者的Java和通用的面向对象编程的能力.下面的章节分为上下两篇,第 ...

  6. c++解析csv 存入数组_使用Apache Commons CSV在Java中读写CSV

    介绍 这是专门针对Java读写CSV的库的简短系列文章的第二篇,也是上一篇文章" Core Java读写CSV"的直接续篇. Apache Commons CSV 在Apache的 ...

  7. Java 面试知识点解析(七)——Web篇

    前言: 在遨游了一番 Java Web 的世界之后,发现了自己的一些缺失,所以就着一篇深度好文:知名互联网公司校招 Java 开发岗面试知识点解析 ,来好好的对 Java 知识点进行复习和学习一番,大 ...

  8. Java 面试知识点解析(六)——数据库篇

    前言: 在遨游了一番 Java Web 的世界之后,发现了自己的一些缺失,所以就着一篇深度好文:知名互联网公司校招 Java 开发岗面试知识点解析 ,来好好的对 Java 知识点进行复习和学习一番,大 ...

  9. JAVA教程 第六讲 Java的线程和Java Applet(二)

    6.3 Java Applet 前面的章节我们阐述了Application的应用,这一讲我们将介绍java的另一类应用java Applet,即java小应用程序. 在Java问世的头几年里,之所以如 ...

最新文章

  1. CSDN上究竟可以上载多大的GIF文件?
  2. GetSystemMetrics()函数的用法
  3. 序列计数(动态规划/自动机/前缀和优化)
  4. wcf编程用什么工具_四个强大的自学编程网站工具,用的人都是学习欲望很强!...
  5. 周末ROS学习沙龙第三期——launch文件、自定义服务通信、控制机器人移动、传感器数据处理
  6. python中面向对象的ui_Python面向对象和图形用户界面(一)---- 面向对象
  7. 视频教程-H3C-H3CNE 华三网络工程师从入门到精通 自学视频课程[肖哥]-H3C认证
  8. Excel如何快速生成二维码图片?
  9. 竹间智能:人机交互未来如何改变人类生活
  10. 数值分析思考题(钟尔杰版)参考解答——第六章
  11. 搜索引擎优化(step-by-step)
  12. TortoiseGit 使用教程
  13. Linux驱动BSP(I2C 驱动实验)
  14. UART协议及串口回环
  15. 区块链和大数据的关系
  16. win7 下anaconda 安装及安装包
  17. 抖音私信名片_抖音消息卡片_抖音跳转微信_抖音私信跳转_抖音落地页
  18. 极客日报:日本的首颗 5nm 芯片公布;虾米音乐正式关停;网易云音乐再次喊话酷狗...
  19. wcs系统安全保护功能
  20. 52.桌面上的IE图标不见了怎么办:

热门文章

  1. 快夸我!我把SpringBoot项目从18.18M瘦身到0.18M!
  2. 一文读懂常用日志框架(Log4j、SLF4J、Logback)有啥区别
  3. 转向AIOps之前,你应该做好哪些准备?
  4. Linux 常用命令全称,看看你 get 到了哪些?
  5. docker化你的java应用(上)
  6. 开机遇到grub解决方法,超详细
  7. spring mvc事务没有生效的原因
  8. Spring Cloud构建微服务架构(六)高可用服务注册中心
  9. Java集合:Hashtable源码分析
  10. 14.线程安全?线程不安全?可重入函数?不可重入函数?