java.applet包的AppletContext类包含如下两个成员方法:getApplet和getApplets。使用这两个方法,Applet能够寻找到其他的Applet并调用它们的方法。不过这是以满足如下安全条件为前提的:
所有Applet位于同一服务器的同一目录。
所有Applet运行在同一页面且位于同一浏览器窗口内。
或许为Applet加上这些安全限制都有着很充分的理由,但第二个条件给我们实现带有Applet到Applet通信功能的多Applet界面带来了限制。
请设想如下情形:
你刚好编写完了一个用于股票交易的Applet,接下来想要为它做一个完善的帮助系统。这个帮助系统也是一个Applet,而且应该放到和股票交易Applet不同的另外一个帧里。你作出这个决定的原因可能是出于网站整体结构的考虑,也可能是你想让帮助系统总是处于显示状态。你还想让帮助Applet能够自动根据用户的当前操作切换到相应的帮助条目(就象Microsoft Office帮助系统提供的那样)。此外,你甚至还计划着在帮助系统里做一个向导,使它能够在另外一个窗口中指导用户如何完成股票交易Applet里的各种任务。
应当承认这种设想是非常美妙的。然而,由于这些Applet位于不同的页面,AppletContext中的Java API无法帮助你完全实现这种设想――不过本文介绍的技术能够。
在介绍新的Applet到Applet通信方法之前,我们先要简要地介绍一下getApplet和getApplets这两个方法的用法。Applet能够寻找到同一页面的其他Applet,这既可以调用getApplet方法通过名字查找,也可以调用getApplets方法找出同一页面内的所有Applet。这两个方法都在调用成功时返回一个或多个Applet对象给调用者。调用者找到目标Applet对象之后,接着就可以调用它的公用方法。
假设HTML页面的代码片断如下:
通过APPLET标记的name属性,我们可以用如下代码引用指定的Applet:
Applet theOtherApplet = getApplet("app1");
//调用该Applet的公用方法
theOtherApplet.anyMethod();
或者,我们也可以用下面的代码提取页面内的所有Applet:
Enumeration allAppletsOnSamePage = getApplets();
while(allAppletsOnSamePage.hasMoreElements()) {
Applet appl = (Applet) allAppletsOnSamePage.nextElement();
// 调用Applet的公用方法
appl.anyMethod();
}
当发出调用的Applet提取到同一HTML页面内的一个或多个Applet对象时,它就可以调用这些Applet对象的公用方法。
遗憾的是,使用这种标准方法我们只能实现同一页面内的Applet通信。但如果幸运的话,我们可以很方便地突破这一局限。实现跨页面Applet到Applet通信基于这样一个事实,即如果两个Applet具有相同的codebase,那么即使它们运行在不同的浏览器窗口中,它们也将共享同一个运行时环境。所谓的codebase,我们可以粗略地把它看成Applet所在的服务器目录,请参见本文后面参考资源中有关codebase示范的链接。
共享运行时环境使得类的静态域和结构能够被所有的Applet实例访问,因此,我们可以用这些静态域和结构在不同的Applet之间传递信息。
我们不仅可以把简单数据类型――比如整数、字符、字符串存储到这些静态域,而且还可以存储Applet实例本身的引用,这样,其他的Applet就可以通过访问这些静态域得到该Applet实例的引用。
这种方法非常复杂吗?事实上,它并不复杂。下面我们来看一个简单的例子。假设有两个Applet(AppletA.class和AppletB.class)位于不同的帧,但它们具有相同的codebase。
现在我们要从AppletA里面访问AppletB的公用方法。首先我们要在AppletB里面把它自己的引用保存到一个静态公用域,如:
public class AppletB {
public static AppletB selfRef = null; // Initially zero
public void init() {
// 保存当前实例的引用
selfRef = this;
}
...
}
现在AppletA可以访问AppletB的实例:
public class AppletA {
AppletB theOtherApplet = null;
public void callAppletB() {
// 获得静态域的值,这个静态域保存了
// AppletB实例的引用
theOtherApplet = AppletB.selfRef;
// 接下来就可以调用AppletB实例
// 的方法,例如:
theOtherApplet.repaint();
}
...
}
这样就实现了两个Applet的通信。由于不同的Applet共享运行时环境,因此即使这些Applet位于不同的页面,这种方法也同样有效。
不过应当注意的是,上面的代码不能处理这种情况:在AppletB没有启动之前就在AppletA里面调用callAppletB方法。如果发生这种情况,则selfRef的值将是null,Applet之间的通信不能正常进行。
当然,我们还可以设计出更加通用的方法。我们可以创建一个类,这个类的唯一用途就是在自己的静态数据结构中保存其他Applet的引用。下面是一个参考实现AppletList。想要让其他Applet访问自己的公用方法的Applet实例首先要在AppletList中注册。按照AppletContext.getApplet(string name)方法的处理模式,每个注册的Applet都和一个字符串相关联。以后当其他Applet需要引用某个Applet实例时,这个字符串就可以作为键(即Applet的标识)使用。
下面是Applet在AppletList中注册的典型过程:
public class AppletA {
public void start() {
AppletList.register("Stock-trade-applet", this);
...
}
}
其他Applet访问已注册Applet的过程如下:
public class AppletB {
public void run() {
AppletA tradeApplet =
(AppletA) AppletList.getApplet("Stock-trade-applet");
...
}
}
当Applet结束运行时它必须从ApplietList取消注册:
public void stop() {
AppletList.remove("Stock-trade-applet");
...
}
AppletList类的完整代码如下:
0: import java.util.*;
1: import java.applet.Applet;
2:
3: public class AppletList {
4: private static Hashtable applets = new Hashtable();
5:
6: public static void register(String name, Applet applet) {
7: applets.put(name,applet);
8: }
9:
10: public static void remove(String name) {
11: applets.remove(name);
12: }
13:
14: public static Applet getApplet(String name) {
15: return (Applet) applets.get(name);
16: }
17:
18: public static Enumeration getApplets() {
19: return applets.elements();
20: }
21:
22: public static int size() {
23: return applets.size();
24: }
25: }
请从本文后面下载exampleCode.zip了解更多有关如何应用AppletList类的信息。
本方法的局限
如前所述,参与通信的Applet必须具有相同的codebase。此外,如果你运行的是两个不同的浏览器副本且Applet分别运行于这两个浏览器中,由于这些Applet可能没有共享运行时环境(这和浏览器版本、设置有关),因此它们可能不能进行通信。然而,如果你是从同一个浏览器创建出新的浏览器窗口,那么这个问题是不存在的。
本文所介绍的技术已经顺利通过好几个平台和浏览器版本的测试,但导致多个Applet拥有各自的运行时环境的配置还是存在的。顺利通过测试的操作系统和浏览器组合如下所示:
小结
本文介绍了一种新的实现Applet到Applet通信的方法,这种方法在无法使用Java API的getApplet()方法时仍旧有效。掌握了这种方法,你就有更多的机会在Web网站或Intranet上使用Applet――用它来替换getApplets方法,或者补充getApplets方法的不足。
所有Applet位于同一服务器的同一目录。
所有Applet运行在同一页面且位于同一浏览器窗口内。
或许为Applet加上这些安全限制都有着很充分的理由,但第二个条件给我们实现带有Applet到Applet通信功能的多Applet界面带来了限制。
请设想如下情形:
你刚好编写完了一个用于股票交易的Applet,接下来想要为它做一个完善的帮助系统。这个帮助系统也是一个Applet,而且应该放到和股票交易Applet不同的另外一个帧里。你作出这个决定的原因可能是出于网站整体结构的考虑,也可能是你想让帮助系统总是处于显示状态。你还想让帮助Applet能够自动根据用户的当前操作切换到相应的帮助条目(就象Microsoft Office帮助系统提供的那样)。此外,你甚至还计划着在帮助系统里做一个向导,使它能够在另外一个窗口中指导用户如何完成股票交易Applet里的各种任务。
应当承认这种设想是非常美妙的。然而,由于这些Applet位于不同的页面,AppletContext中的Java API无法帮助你完全实现这种设想――不过本文介绍的技术能够。
在介绍新的Applet到Applet通信方法之前,我们先要简要地介绍一下getApplet和getApplets这两个方法的用法。Applet能够寻找到同一页面的其他Applet,这既可以调用getApplet方法通过名字查找,也可以调用getApplets方法找出同一页面内的所有Applet。这两个方法都在调用成功时返回一个或多个Applet对象给调用者。调用者找到目标Applet对象之后,接着就可以调用它的公用方法。
假设HTML页面的代码片断如下:
通过APPLET标记的name属性,我们可以用如下代码引用指定的Applet:
Applet theOtherApplet = getApplet("app1");
//调用该Applet的公用方法
theOtherApplet.anyMethod();
或者,我们也可以用下面的代码提取页面内的所有Applet:
Enumeration allAppletsOnSamePage = getApplets();
while(allAppletsOnSamePage.hasMoreElements()) {
Applet appl = (Applet) allAppletsOnSamePage.nextElement();
// 调用Applet的公用方法
appl.anyMethod();
}
当发出调用的Applet提取到同一HTML页面内的一个或多个Applet对象时,它就可以调用这些Applet对象的公用方法。
遗憾的是,使用这种标准方法我们只能实现同一页面内的Applet通信。但如果幸运的话,我们可以很方便地突破这一局限。实现跨页面Applet到Applet通信基于这样一个事实,即如果两个Applet具有相同的codebase,那么即使它们运行在不同的浏览器窗口中,它们也将共享同一个运行时环境。所谓的codebase,我们可以粗略地把它看成Applet所在的服务器目录,请参见本文后面参考资源中有关codebase示范的链接。
共享运行时环境使得类的静态域和结构能够被所有的Applet实例访问,因此,我们可以用这些静态域和结构在不同的Applet之间传递信息。
我们不仅可以把简单数据类型――比如整数、字符、字符串存储到这些静态域,而且还可以存储Applet实例本身的引用,这样,其他的Applet就可以通过访问这些静态域得到该Applet实例的引用。
这种方法非常复杂吗?事实上,它并不复杂。下面我们来看一个简单的例子。假设有两个Applet(AppletA.class和AppletB.class)位于不同的帧,但它们具有相同的codebase。
现在我们要从AppletA里面访问AppletB的公用方法。首先我们要在AppletB里面把它自己的引用保存到一个静态公用域,如:
public class AppletB {
public static AppletB selfRef = null; // Initially zero
public void init() {
// 保存当前实例的引用
selfRef = this;
}
...
}
现在AppletA可以访问AppletB的实例:
public class AppletA {
AppletB theOtherApplet = null;
public void callAppletB() {
// 获得静态域的值,这个静态域保存了
// AppletB实例的引用
theOtherApplet = AppletB.selfRef;
// 接下来就可以调用AppletB实例
// 的方法,例如:
theOtherApplet.repaint();
}
...
}
这样就实现了两个Applet的通信。由于不同的Applet共享运行时环境,因此即使这些Applet位于不同的页面,这种方法也同样有效。
不过应当注意的是,上面的代码不能处理这种情况:在AppletB没有启动之前就在AppletA里面调用callAppletB方法。如果发生这种情况,则selfRef的值将是null,Applet之间的通信不能正常进行。
当然,我们还可以设计出更加通用的方法。我们可以创建一个类,这个类的唯一用途就是在自己的静态数据结构中保存其他Applet的引用。下面是一个参考实现AppletList。想要让其他Applet访问自己的公用方法的Applet实例首先要在AppletList中注册。按照AppletContext.getApplet(string name)方法的处理模式,每个注册的Applet都和一个字符串相关联。以后当其他Applet需要引用某个Applet实例时,这个字符串就可以作为键(即Applet的标识)使用。
下面是Applet在AppletList中注册的典型过程:
public class AppletA {
public void start() {
AppletList.register("Stock-trade-applet", this);
...
}
}
其他Applet访问已注册Applet的过程如下:
public class AppletB {
public void run() {
AppletA tradeApplet =
(AppletA) AppletList.getApplet("Stock-trade-applet");
...
}
}
当Applet结束运行时它必须从ApplietList取消注册:
public void stop() {
AppletList.remove("Stock-trade-applet");
...
}
AppletList类的完整代码如下:
0: import java.util.*;
1: import java.applet.Applet;
2:
3: public class AppletList {
4: private static Hashtable applets = new Hashtable();
5:
6: public static void register(String name, Applet applet) {
7: applets.put(name,applet);
8: }
9:
10: public static void remove(String name) {
11: applets.remove(name);
12: }
13:
14: public static Applet getApplet(String name) {
15: return (Applet) applets.get(name);
16: }
17:
18: public static Enumeration getApplets() {
19: return applets.elements();
20: }
21:
22: public static int size() {
23: return applets.size();
24: }
25: }
请从本文后面下载exampleCode.zip了解更多有关如何应用AppletList类的信息。
本方法的局限
如前所述,参与通信的Applet必须具有相同的codebase。此外,如果你运行的是两个不同的浏览器副本且Applet分别运行于这两个浏览器中,由于这些Applet可能没有共享运行时环境(这和浏览器版本、设置有关),因此它们可能不能进行通信。然而,如果你是从同一个浏览器创建出新的浏览器窗口,那么这个问题是不存在的。
本文所介绍的技术已经顺利通过好几个平台和浏览器版本的测试,但导致多个Applet拥有各自的运行时环境的配置还是存在的。顺利通过测试的操作系统和浏览器组合如下所示:
小结
本文介绍了一种新的实现Applet到Applet通信的方法,这种方法在无法使用Java API的getApplet()方法时仍旧有效。掌握了这种方法,你就有更多的机会在Web网站或Intranet上使用Applet――用它来替换getApplets方法,或者补充getApplets方法的不足。