Java 打包准则

来自 Arch Linux 中文维基
Arch 打包准则

32 位CLRCMakeCrossDKMSEclipseElectronFontFree PascalGNOMEGoHaskellJavaKDEKernelLispMesonMinGWNode.jsNonfreeOCamlPerlPHPPythonRRubyRustVCSWebWine

本文档定义了 Arch Linux 下打包 Java 程序的建议标准。Java 程序是出了名的难以在没有重叠的依赖关系的情况下干净利落地打包。本文档描述了一种处理这种情况的方法。这些指南十分灵活,以涵盖处理 Java 程序时遇到的许多不同情况。

介绍[编辑 | 编辑源代码]

Arch Linux 的打包人员似乎无法就如何处理 Java 包达成一致。官方和非官方软件库以及 AURPKGBUILD 中使用了各种方法。这包括把所有东西扔到 /opt 里,把 Shell 脚本放在 /usr/bin 或把配置文件放在 /etc/profile 中。其他的则是放在 /usr/share 目录中,脚本放在 /usr/bin 中。许多做法在系统 CLASSPATHPATH 中添加了不必要的文件。

典型 Java 应用程序的结构[编辑 | 编辑源代码]

大多数桌面 Java 应用程序结构类似。它们是由一个不依赖系统(但依赖软件包!)的安装程序安装的。这通常会把所有东西都安装在包含 binlibjarconf等子目录的目录下。一般会有一个包含主要可执行类的主 jar 文件。通常还会提供一个 Shell 脚本来运行主类,这样用户就不必直接调用 Java 解释器。这个 Shell 脚本通常相当复杂,因为它需要在不同的发行版中通用,而且经常包括不同系统的特殊情况(例如 Cygwin)。

lib 目录通常包含捆绑的 jar 文件来满足 Java 程序的依赖。这使用户安装程序变得简单(其中包含了所有依赖),但却是软件包开发者的噩梦。几个软件包捆绑相同的依赖会浪费空间。这在过去并不是一个大问题,当时桌面 Java 应用程序和库较少,而那些存在的应用程序和库本身往往也非常大。但现在情况不同了……

运行程序所需的其他文件通常与主 jar 文件存放在同一目录或其子目录下。由于 Java 程序不知道它们的类是从哪里加载的,所以通常需要在这个目录中运行(即 Shell 脚本应 cd 进入该目录),或者设置一个环境变量来指示这个目录的位置。

在 Arch Linux 中打 Java 包[编辑 | 编辑源代码]

在 Arch 中打包 Java 应用程序将会比现在更费劲。然而,这是值得的,它可以使文件系统更干净,捆绑的依赖更少(随着越来越多的Java 库被重构为单独的包,打包会变得更容易)。在创建 Arch Linux Java 包时,应遵循以下准则:

  • 如果 Java 库有一个通用的名字,那么包的名字应该在前面加上 java- 以与其他库区分。对于命名独特的包(如 JUnit)、终端用户程序(如 Eclipse)或可以用其他前缀唯一描述的库(如 jakarta-commons-collections 或 apache-ant),则不需要这样做。
  • 将所有随程序分发的 jar 文件(而不是其他文件)放在 /usr/share/java/程序目录下。这包括随程序分发的所有依赖 jar 文件。然而,应该努力将常见的或大型的依赖放到单独的包中。这只有在程序不依赖某个特定版本的库时才能实现。
这个规则使得迭代重构依赖关系成为可能。也就是说,软件包和它的所有依赖可以先放在一个目录中。经过测试后,主要的依赖可以一个一个地被重构出来。请注意,有些应用程序在主 jar 文件中包括捆绑的依赖。也就是说,他们将捆绑的依赖解包,并将其包含在主 jar 中。这样的依赖通常非常小,重构它们没有什么意义。
  • 如果程序是要由用户运行的,编写一个自定义的 Shell 脚本运行主 jar 文件。这个脚本应该放在 /usr/bin 中。库一般不需要 Shell 脚本。从头开始编写 Shell 脚本,不要使用程序捆绑的脚本。删除检查自定义环境(如 Cygwin)的代码,以及试图确定 JAVA_HOME 是否已经设置的代码(Arch 不使用 JAVA_HOME,而是使用 archlinux-java 来设置 /usr/bin/java 符号连接)。
寻找 jar 文件的脚本应该类似这样:
#!/bin/sh
exec /usr/bin/java -jar '/usr/share/java/程序/程序.jar' "$@"
寻找单个类文件的应该类似这样:
#!/bin/sh
exec /usr/bin/java '/usr/share/java/程序/程序类名' "$@"
  • 使用 -cp 选项为 Java 解释器设置 CLASSPATH,除非有明确的理由不这样做(即:CLASSPATH 用于插件机制)。CLASSPATH 应该包括 /usr/share/java/程序目录下的所有 jar 文件,以及已被重构到其他目录中的依赖库的 jar 文件。可以使用类似以下代码:
for name in /usr/share/java/程序/*.jar ; do
  CP=$CP:$name
done
CP=$CP:/usr/share/java/dep1/dep1.jar
java -cp $CP 程序.java.MainClass
  • 确保 Shell 脚本可执行!
  • 随软件包一起分发的其他文件应存放在 /usr/share 下以该软件包命名的目录中。你可能需要在 Shell 脚本中的 项目_HOME 等变量中设置该目录的位置。本准则假设程序希望所有文件都在同一目录下(这是 Java 软件包的标准做法)。如果配置文件更适合放在其他地方(例如,把日志放在 /var/log 中),则可以这样做。
请记住,在某些系统上 /usr 可能被挂载为只读。如果应用程序需要写入共享目录中的文件,它们可能需要被重新定位到/etc/var 或用户的主目录。
  • 与其他 Arch Linux 软件包的标准一样,如果遵守上述标准需要做大量的工作,则应以其偏好的方式安装软件包,并将生成的目录放在 /opt 中。这对于捆绑 JRE、包含自定义版本的依赖或做其他奇怪和痛苦的事情的程序很有用。

多个API实现[编辑 | 编辑源代码]

如果软件包分发了常用的 API 实现(比如 jdbc 驱动),则应该把库放在 /usr/share/java/API_名称 下。这样,允许用户选择实现的应用程序会知道在哪里寻找它们。这个位置只用于库包本身。如果这样的实现是应用程序分发的一部分,不要将这个 jar 文件放在公共位置,而是使用普通的包结构。

目录结构示例[编辑 | 编辑源代码]

为了说明问题,下面是一个假想的名为 foo 程序的目录结构示例。由于 foo 是一个常见的名字,所以软件包被命名为 java-foo,但注意目录结构中并没有体现:

  • /usr/share/java/foo/
  • /usr/share/java/foo/foo.jar
  • /usr/share/java/foo/bar.jarjava-foo 包含的依赖)
  • /usr/share/foo/
  • /usr/share/foo/*.*java-foo 需要的一些文件)
  • /usr/bin/foo(可执行的 Shell 脚本)

依赖[编辑 | 编辑源代码]

根据需要,Java 软件包可能会依赖 java-runtimejava-environment

对于大多数软件包来说,运行用 Java 编写的软件只需要 java-runtime

需要将 Java 源代码编译到字节码的软件包需要 java-environment(如 JDK)。

更多信息参见 Java