这些Java教程是针对JDK 8编写的。本页面中描述的示例和实践不利用后续版本中引入的改进,并且可能使用不再可用的技术。
有关Java SE 9及后续版本中更新的语言特性的概述,请参阅Java语言更改。
有关所有JDK版本的新功能、增强功能以及已删除或弃用选项的信息,请参阅JDK发行说明。
JarRunner应用程序是以以下形式的命令启动的:
java JarRunner url [arguments]
在前一节中,我们已经看到JarClassLoader如何能够从给定的URL识别和加载JAR捆绑应用程序的主类。因此,为了完成JarRunner应用程序,我们需要能够从命令行获取URL和任何参数,并将它们传递给JarClassLoader的实例。这些任务属于JarRunner类,即JarRunner应用程序的入口点。
它首先通过URL上指定的命令行参数创建一个java.net.URL对象:
public static void main(String[] args) {
if (args.length < 1) {
usage();
}
URL url = null;
try {
url = new URL(args[0]);
} catch (MalformedURLException e) {
fatal("Invalid URL: " + args[0]);
}
如果args.length < 1,这意味着命令行中没有指定URL,所以会打印一个用法消息。如果第一个命令行参数是一个有效的URL,就会创建一个新的URL对象来表示它。
接下来,JarRunner创建一个新的JarClassLoader实例,将命令行上指定的URL传递给构造函数:
JarClassLoader cl = new JarClassLoader(url);
正如我们在前一节中看到的,正是通过JarClassLoader,JarRunner才能接入JAR处理API。
传递给JarClassLoader构造函数的URL是您要运行的JAR捆绑应用程序的URL。接下来,JarRunner调用类加载器的getMainClassName方法来识别应用程序的入口类:
String name = null;
try {
name = cl.getMainClassName();
} catch (IOException e) {
System.err.println("I/O error while loading JAR file:");
e.printStackTrace();
System.exit(1);
}
if (name == null) {
fatal("Specified jar file does not contain a 'Main-Class'" +
" manifest attribute");
}
加粗的语句是关键语句。其他语句用于错误处理。
一旦JarRunner识别出应用程序的入口类,只剩下两个步骤:将任何参数传递给应用程序并实际启动应用程序。JarRunner通过以下代码执行这些步骤:
// Get arguments for the application
String[] newArgs = new String[args.length - 1];
System.arraycopy(args, 1, newArgs, 0, newArgs.length);
// Invoke application's main class
try {
cl.invokeClass(name, newArgs);
} catch (ClassNotFoundException e) {
fatal("Class not found: " + name);
} catch (NoSuchMethodException e) {
fatal("Class does not define a 'main' method: " + name);
} catch (InvocationTargetException e) {
e.getTargetException().printStackTrace();
System.exit(1);
}
请记住,第一个命令行参数是JAR捆绑应用程序的URL。要传递给该应用程序的任何参数都在args数组的元素1及以后。JarRunner将这些元素取出,并创建一个名为newArgs的新数组来传递给应用程序(上面加粗的行)。然后,JarRunner将入口类的名称和新的参数列表传递给JarClassLoader的invokeClass方法。正如我们在前一节中看到的,invokeClass将加载应用程序的入口类,传递任何参数,并启动应用程序。