下一代Java Applet插件技术(译)
下一代Java Applet插件技术
Java SE 6对Java桌面应用进行较大的升级,并启动了Java SE 6 Update N计划,该计划旨在简化JRE的大小,增进用户的安装体验,并提供了一个新的Applet浏览器插件,该插件将会随Java SE 6 Update 10发布。本文全面介绍了这个新插件的关键特性,并以NASA的World Wind为例介绍了该插件的应用。(2008.07.15最后更新)Applet回来了!
为了在网络中传递你的程序,是时候再次考虑Java Applet技术了。下一代Java插件技术以一种不同的,比过去更高效、更可靠的途径来运行Applet。现在你可以获得如下好处:
- 增强的可靠性
- 改进的用户体验
- 在后台启动Applet
- 内建的JNLP支持
- 针对每个Applet的命令行参数
- 堆内存大小,Java 2D API加速选项
- 改进的Java/JavaScript程序设计语言集成
- 改进的Windows Vista支持
- 签名的Applet现在可以在Internet Explorer的保护模式中正常运行
下一代Java插件最有意义的新特性是它内建支持通过JNLP文件启动Applet。使用JNLP文件格式作为Applet的描述符就能允许Applet马上复用之前为Java Web Start应用所写的JNLP扩展。
执行Applet的新途径
执行Applet的新途径在结构上与Java Web Start技术相似,但与浏览器整合的更为紧密。Applet不在运行于网络浏览器内的JVM中,而是会启动一个独立的JVM进程去运行Applet。默认地,只有一个JVM将被启动,但你也能启动多个JVM,并且可以为每个Applet都设置命令行参数,所以你能影响堆内存的大小或其它的要求。
Figure 1. Applet Architecture
在上图中,云表示JVM实例。在浏览器内有一个小的,headless JVM被用于管理一个或多个客户端JVM之间的连接,这些JVM运行着Applet。在该图中,Duke表示Applet。其中,一个JVM实例运行着两个Applet,另一个运行着一个Applet。
Applet直接从JNLP文件启动,它使用的JNLP文件与Java Web Start软件使用的描述符文件相同,并且允许使用比典型的"archive","code"和"cache_archive"更为强大的参数。
新的插件提供了:
- 能够访问之前仅由Java Web Start软件专用的高级JNLP扩展。之前有少部分参数能够使用,但有一些限制,现在这些限制则被去除。
- 通过Applet访问JNLP API。
- 支持PersistenceService和DownloadService。
- 能够控制堆内存大小,命令行参数,JRE版本选择和自动下载。你具有Java Web Start软件所拥有的相同功能。
<
applet width
=
”
500
” height
=
”
500
”
>
< param name = ”jnlp_href” value = ”my_applet.jnlp” >
</ applet >
调用Applet生命周期方法init,start,stop和destroy会更为确定,并且已经改进了跨浏览器行为。完全支持Applet类装载器缓存,遗留的Applet生命周期及对向后兼容性的需求,并且这些行为都已得到了改进。< param name = ”jnlp_href” value = ”my_applet.jnlp” >
</ applet >
Applet运行的就像一个由Java Web Start启动的应用。参数jnlp_href在Web页面和Applet的JNLP描述之间起到了桥接的作用。在如宽度与高度这样的特定方面,Applet标签与JNLP文件具有重叠的机制。
一般地,你应该使用Deployment Toolkit,这也是一个出现在Java SE 6 Update 10中的新工具,它能自动地为Applet标签生成HTML。部署建议指南展示了如何使用Deployment Toolkit简便地发布Applet。
配置Applet
现在也能更为简单地在多个方面来配置Applet,包括堆内存大小,需要被使用的Java版本,类加载器缓存,边界,及其它。
<applet>与JNLP文件在针对某些参数时有重叠的机制。这些冲突可以用如下方法解决:
- width and height:这些属性将总是从<applet>,而不是JNLP文件,中获取。这是假设浏览器知道Applet在Web页面上应该显示多大,也只有浏览器才能支持相对于页面的宽度与高度(例如,width="50%")。
- codebase:如果JNLP文件在<jnlp>标签中指定了一个绝对的codebase,那么就使用它。否则,将使用在codebase handling一节中描述的规则进行组织。
- code:当指定了jnlp_href参数,Applet的主类名将从main-class参数换成JNLP文件中的applet-desc标签,并且code属性会被忽略。注意,该特性允许你为经典Java插件写一个拥有反馈的Applet标签,但在新的Java插件中,该标签可使用更高级的功能。请见下面的"兼容性"一节。
- 任何一个由<param>标签指定的Applet参数将与JNLP文件中指定的参数进行合并。如果<applet>标签和JNLP文件都指定了同一个参数,<applet>标签中的版本将覆盖JNLP文件中的版本,除了参数java_arguments和java_version。
- 新的java_arguments和java_version参数在JNLP Applet中是不必要的。会替换为通过JNLP文件请求JRE版本或向JVM传递参数的机制。所以,命令行参数和JNLP文件请求的JRE版本将会覆盖HTML中为Applet指定的这些值。
- 特定的参数,例如image,boxbgcolor等等,在Applet的启动过程中是有用的。在HTML而不是JNLP文件中指定这些参数可能更好些,以便于在加载Web页面时就可立即获取它们,而不用再等到单独下载JNLP文件之后。
指定一个比默认值大的堆空间:
<
APPLET archive
=
"
my_applet.jar
"
code
=
"
MyApplet
"
width
=
"
300
"
height
=
"
300
"
>
< PARAM name = " java_arguments " value = " -Xmx128m " >
</ APPLET >
指定一个非默认大小的堆内存以及一个Java 2D硬件加速器选项,该选项常通过JOGL使用OpenGL应用于Applet。< PARAM name = " java_arguments " value = " -Xmx128m " >
</ APPLET >
<
APPLET archive
=
"
my_applet.jar
"
code
=
"
MyApplet
"
width
=
"
300
"
height
=
"
300
"
>
< PARAM name = " java_arguments " value = " -Xmx256m -Dsun.java2d.noddraw=true " >
</ APPLET >
如果你喜欢,一个Applet可强制进入一个属于它自己的JVM实例,而与所有其它的Applet隔离开:< PARAM name = " java_arguments " value = " -Xmx256m -Dsun.java2d.noddraw=true " >
</ APPLET >
<
param name
=
”separate_jvm” value
=
”
true
”
/>
当把某些桌面应用移植到Web浏览器时,这就有用了。你也能使特定的Applet运行在特定版本的JRE上,如下所示:
<
j2se version
=
”
1.4
+
”
>
< j2se version = ” 1.5 * ” >
当想针对特定版本的JRE,或Applet取代早期版本的选择机制(如同IE浏览器中的CLSID),进行质量测评时,该方法就很有用了。如果请求了一个非常老的JRE版本,就会强制执行限制;如果Applet试图加载未签名的代码,将会提示用户。< j2se version = ” 1.5 * ” >
注意,因为支持JNLP的Java插件是在Java SE 6 Update 10中才首次出现的,所以指定像“1.4+”这样的版本基本上没有意义的。当需要“1.7+”时,这才有意义。
另外,你可以在JNLP文件中使用<update>标签来显著降低第二次及接下来各次启动的时间。
<
update check
=
”background”
>
在这种情况下,将使用缓存中已有的Applet程序,并且在后台下载该应用的更新版本。在下次启动时,就会使用新版本。新的插件也能更好地对图像进行定制,在Applet被加载之前会展示该图像。image参数会以支持动画GIF文件作为目标,Java Plug-in Developers' Guide的Special Attributes一节对此有描述。此外,现也支持如下新的参数:
boxborder
一个布尔型参数,用于指定在Applet被加载之前是否在Applet区域的边缘绘制一个宽度一象素的边框。默认为true。我们建议将该值设置为false,特别是将一个动画GIF用作加载期图像时,以避免可能的闪烁。
centerimage
一个布尔型参数,用于指定是否将加载期图像在Applet区域内居中显示,而不是从左上角起始。默认为false。
使用参数boxborder和centerimage的例子:
<
APPLET archive
=
"
large_archive.jar
"
code = " MyApplet "
width = " 300 " height = " 300 " >
<!-- Use an animated GIF as an indeterminate progress bar
while the applet is loading -->
< PARAM NAME = " image " VALUE = " animated_gif.gif " >
<!-- Turn off the box border for better blending with the
surrounding web page -->
< PARAM NAME = " boxborder " VALUE = " false " >
<!-- Center the image in the applet's area -->
< PARAM NAME = " centerimage " VALUE = " true " >
</ APPLET >
code = " MyApplet "
width = " 300 " height = " 300 " >
<!-- Use an animated GIF as an indeterminate progress bar
while the applet is loading -->
< PARAM NAME = " image " VALUE = " animated_gif.gif " >
<!-- Turn off the box border for better blending with the
surrounding web page -->
< PARAM NAME = " boxborder " VALUE = " false " >
<!-- Center the image in the applet's area -->
< PARAM NAME = " centerimage " VALUE = " true " >
</ APPLET >
兼容性
现在可更容易维护向后兼容性。你可创建运行在更早Java插件版本上的程序,但仅需提供一个与jnlp_href参数一样的格式完整的<applet>标签就可使用这些新特性了。早期版本的JRE会忽略jnlp_href参数,转而使用<applet>标签。新的Java插件技术将忽略archive和code参数,而仅使用JNLP文件去启动Applet。
World Wind Applet示例
由World Wind Java开发组创建的NASA World Wind Java Applet示例阐述了如果发布像NASA World Wind Java这样的领先类库。同样地,也用示例说明了如何使用JavaScript在Web页面中高效地整合HTML和Applet内容。
Figure 2. NASA World Wind Applet
该Web页面包含了关于喀斯喀特山脉的信息(要感谢维基百科),并且将World Wind Java作为一个Applet嵌入其中,以图示该山脉中各山的位置。
<
applet id
=
"
wwjApplet
"
width
=
600
height
=
380
code = " gov.nasa.worldwind.examples.applet.WWJApplet "
archive = " BackwardCompatibility.jar " >
< param name = " jnlp_href " value = " WWJApplet.jnlp " >
</ applet >
WWJApplet随标准的World Wind Java发行包发布。如下所述,你可选择编写你自己的Applet类并将World Wind嵌入其中:code = " gov.nasa.worldwind.examples.applet.WWJApplet "
archive = " BackwardCompatibility.jar " >
< param name = " jnlp_href " value = " WWJApplet.jnlp " >
</ applet >
下面是WWJApplet.jnlp文件中相关的部分:
<
jnlp href
=
"
WWJApplet.jnlp
"
>
< resources os = " Windows " >
< property name = " sun.java2d.noddraw " value = " true " />
</ resources >
< resources >
< j2se href = " http://java.sun.com/products/autodl/j2se " version = " 1.4+ " />
< jar href = " worldwind.jar " main = " true " />
< extension name = " jogl "
href = " http://download.java.net/media/jogl/builds/archive/jsr-231-webstart-current/jogl.jnlp " />
</ resources >
< applet - descname = " WWJ Applet "
main - class = " gov.nasa.worldwind.examples.applet.WWJApplet "
<!-- Overwritten by the surrounding web page -->
width = " 100 "
height = " 100 " >
</ applet - desc >
</ jnlp >
注意几点:< resources os = " Windows " >
< property name = " sun.java2d.noddraw " value = " true " />
</ resources >
< resources >
< j2se href = " http://java.sun.com/products/autodl/j2se " version = " 1.4+ " />
< jar href = " worldwind.jar " main = " true " />
< extension name = " jogl "
href = " http://download.java.net/media/jogl/builds/archive/jsr-231-webstart-current/jogl.jnlp " />
</ resources >
< applet - descname = " WWJ Applet "
main - class = " gov.nasa.worldwind.examples.applet.WWJApplet "
<!-- Overwritten by the surrounding web page -->
width = " 100 "
height = " 100 " >
</ applet - desc >
</ jnlp >
- 在本例中,worldwind.jar作为主类使用。理想地,从NASA的网站引用它,将其作为一个JNLP扩展,这就使得许多不同的都嵌入了World Wind的应用程序或Applet共享相同的jar文件。详情请见下面的内容。
- 为了它的硬件加速的3D图形,World Wind Java使用了针对OpenGL API的Java绑定,JOGL。注意,JOGL JNLP扩展仅使用一行代码与应用程序进行结合。也要注意,在Windows平台上,由于OpenGL API与DirectDraw/Direct3D API(该API用于Windows平台默认的Java 2D实现)之间在驱动层面的冲突,需要指定系统参数-Dsun.java2d.noddraw=true。Windows平台上所有使用JOGL的应用程序与Applet程序都需要该系统参数。
<
a href
=
"
javascript:gotoLocation(MOUNT_RAINIER);
"
>
Mount Rainier
</
a
>
(southeast of Tacoma, Washington)
当点击该链接后,将会调用JavaScript函数gotoLocation。该函数定义在同一个Web页面中:(southeast of Tacoma, Washington)
function
gotoLocation(locationString) {
var params = locationString.split(';');
if (params.length == 3 ) // Lat/lon
getWWJApplet().gotoLatLon(parseFloat(params[ 1 ]),
parseFloat(params[ 2 ]));
}
Web页面HTML中的山峰位置将被解码为JavaScript字符串。将从这些字符串中解析出纬度,经度及其它视觉信息,并将它们传递给Applet。 gotoLatLon方法是在WWJApplet类中定义的;上面的方法调用将起动一个JavaScript-to-Java调用,把参数从JavaScript引擎传给Java。World Wind Applet接收该通知,并将视点以动画的方式切换到适当的地方。注意,gotoLatLon方法会迅速地返回,以便浏览器不必等待它的完成;该动画会在一个单独的Java线程中运行。var params = locationString.split(';');
if (params.length == 3 ) // Lat/lon
getWWJApplet().gotoLatLon(parseFloat(params[ 1 ]),
parseFloat(params[ 2 ]));
}
Figure 3. World Wind Applet with Mount St. Helen's Clicked
如上所述,将World Wind Java集成到你的应用程序或Applet程序中的最好方法是将其作为一个JNLP扩展。这允许很多来自网络的集成了World Wind Java的应用程序或Applet程序能够共享World Wind代码资源。为了引用World Wind JNLP扩展,你需将下面的语句行加入到你的应用程序或Applet程序的JNLP文件中的<resources>部分:
<
extension name
=
"
worldwind
"
href
=
"
http://worldwind.arc.nasa.gov/java/0.4.1/webstart/worldwind.jnlp
"
/>
< extension name = " jogl "
href = " http://download.java.net/media/jogl/builds/archive/jsr-231-webstart-current/jogl.jnlp " />
注意,World Wind扩展JNLP是区分版本的,所以你需参考World Wind文档或访问论坛去找到你的JNLP会引用到的扩展的最新版本。World Wind Central是一个关于World Wind最新信息的有用资源。< extension name = " jogl "
href = " http://download.java.net/media/jogl/builds/archive/jsr-231-webstart-current/jogl.jnlp " />
将World Wind作为一个扩展使用就意味着你不能将WWJApplet直接作为你的main-class使用。由于JNLP文件格式的语义,主jar (main="true")必须定义在主JNLP文件中。但很容易就能适应该限制,你可简单地创建你自己的WWJApplet子类(称之为MyWWJApplet),而它并不做任何事情:
class MyWWJApplet extends WWJApplet {}
将worldwind.jar置于classpath中,并编译上述类,然后将该类放入它自己的jar文件中。引入这个jar作为你的主jar,MyWWJApplet就成为了你的main-class,然后将其作为JNLP扩展引入到World Wind中。结论
介绍了Java插件对JNLP的支持,这为Applet的发布提供了很多新的可能,这对在浏览器内外发布Java内容的方法的统一又进了一大步。Applet自从它们起始已过了很长的时间,现在随着对JNLP的支持,它们会比以往更快,也更易于定制。