当前位置: 首页 > 知识库问答 >
问题:

JLink不产生可重新分发的映像

井昊乾
2023-03-14

有一段时间,我一直在使用模块化项目,但由于受到文件名和自动模块的限制,我从未有机会使用jlink工具来生成可再分发的应用程序映像。今天,我选择启动一个独立的项目,该项目不导入任何外部依赖项,以防止使用兼容性模式。该项目由3个模块组成,并且是maven中的,所以我将只发布我使用的jlink命令片段。

参考项目:https://gitlab.com/dragas/edu-day-demo,签出modules-full标记。项目是用目标构建的,以防止污染本地的。m2存储库。Project已经配置为拉动依赖项,因此打包和部署将更加容易。

我用来生成jlinked图像的命令如下:

jlink \
   --module-path edu-day-runtime/target/dependency/:edu-day-runtime/target/ \
   --add-modules ALL-MODULE-PATH \
   --output edu-day-jlinked \
   --launcher edurun=edu.day.runtime
edu-day-jlinked/bin/edurun 1 1
Result of sum is 2

同时,尝试在容器化环境中运行相同的程序(这里我使用bash:5,一个非java映像来模拟没有安装java的环境)不会产生类似的结果。相反,shell似乎没有找到名为java的二进制文件

docker run -it -v "$(pwd)/edu-day-jlinked:/app" bash:5
...(in container)
bash-5.0# /app/bin/edurun 1 1
/app/bin/edurun: line 4: /app/bin/java: not found

经过检查,该文件夹确实包含名为java的二进制文件

bash-5.0# ls -la
total 52
drwxr-xr-x    2 1000     1000          4096 Aug 23 07:53 .
drwxr-xr-x    7 1000     1000          4096 Aug 23 07:53 ..
-rwxr-xr-x    1 1000     1000           116 Aug 23 07:53 edurun
-rwxr-xr-x    1 1000     1000         16688 Aug 23 07:53 java
-rwxr-xr-x    1 1000     1000         16712 Aug 23 07:53 keytool

但是即使直接调用它(以显示帮助消息)也不会产生任何结果,除了无法找到二进制文件的相同消息之外

(in /app/bin/ folder)
bash-5.0# ./java 
bash: ./java: No such file or directory
(in /app/bin/ folder)
bash-5.0# ./keytool
bash: ./keytool: No such file or directory

共有1个答案

湛联
2023-03-14

您的问题是测试容器(BASH:5)没有使用与java环境相同版本的运行时链接器。

只有当系统上有兼容的linux运行时链接器时,jlink生成的二进制文件才会运行。

运行时链接器的目的是配置二进制文件以便在系统上执行-在构建可执行文件时,默认的运行时链接器被硬编码到二进制文件中。您可以使用诸如readelf-lldd之类的工具检查运行时链接器(只有当ldd能够找到运行时链接器时,ldd才起作用)

i386 linux的默认运行时链接器是:/lib/ld-linux.so.2

bash:5容器中,默认的运行时链接器是:/lib/ld-musl-x86_64.so.1

这与jdk的运行时链接器不兼容

$ docker run --rm -it -v (pwd)/edu-day-jlinked64:/app -w /here bash:5 bash

bash-5.0# /app/bin/java
bash: /app/bin/java: No such file or directory
bash-5.0# strings -a /app/bin/java | grep '^/lib'
/lib64/ld-linux-x86-64.so.2
bash-5.0# ls -l /lib64/ld-linux-x86-64.so.2
ls: /lib64/ld-linux-x86-64.so.2: No such file or directory
bash-5.0# /lib/ld-musl-x86_64.so.1 --list /app/bin/java
/lib64/ld-linux-x86-64.so.2 (0x7fe2852a3000)
libjli.so => /app/bin/../lib/libjli.so (0x7fe28528c000)
libc.so.6 => /lib64/ld-linux-x86-64.so.2 (0x7fe2852a3000)
libz.so.1 => /lib/libz.so.1 (0x7fe285272000)
libdl.so.2 => /lib64/ld-linux-x86-64.so.2 (0x7fe2852a3000)
libpthread.so.0 => /lib64/ld-linux-x86-64.so.2 (0x7fe2852a3000)
Error relocating /app/bin/../lib/libjli.so: __snprintf_chk: symbol not found
Error relocating /app/bin/../lib/libjli.so: __vfprintf_chk: symbol not found
Error relocating /app/bin/../lib/libjli.so: __read_chk: symbol not found
Error relocating /app/bin/../lib/libjli.so: __memmove_chk: symbol not found
Error relocating /app/bin/../lib/libjli.so: __printf_chk: symbol not found
Error relocating /app/bin/../lib/libjli.so: __fprintf_chk: symbol not found
Error relocating /app/bin/../lib/libjli.so: __sprintf_chk: symbol not found

所以在这里肯定行不通。

让我们用一些‘标准’的东西。因为我在Ubuntu:Focal容器中构建了jlinked应用程序,所以我们使用一个没有java内置的java版本:

$ docker run --rm -it -v $(pwd)/edu-day-jlinked64:/app -w /here ubuntu:focal bash
root@865c9c12c029:/here# /app/bin/java
Usage: java [options] <mainclass> [args...]
           (to execute a class)
   or  java [options] -jar <jarfile> [args...]
           (to execute a jar file)
   or  java [options] -m <module>[/<mainclass>] [args...]
       java [options] --module <module>[/<mainclass>] [args...]
           (to execute the main class in a module)
   or  java [options] <sourcefile> [args]
           (to execute a single source-file program)

所以在这种情况下会起作用。

$ docker run --rm -it -v $(pwd):/here -w /here ubuntu:focal bash

# apt-get update
# DEBIAN_FRONTEND=noninteractive apt-get install -y git openjdk-14-jdk maven
# git clone https://gitlab.com/Dragas/edu-day-demo .
# git checkout modules-full
# ./mvnw package
# rm -rf edu-day-runtime/target/classes
# jlink --module-path edu-day-runtime/target/dependency/:edu-day-runtime/target/ --add-modules ALL-MODULE-PATH --output edu-day-jlinked    --launcher edurun=edu.day.runtime
# ./edu-day-jlinked/bin/edurun 1 1
Result of sum is 2
$ docker run --rm -it -v $(pwd)/edu-day-jlinked:/app -w /here bash:5 bash
bash-5.0# /app/bin/edurun 1 1
/app/bin/edurun: line 4: /app/bin/java: not found
$ docker run --rm -it -v $(pwd)/edu-day-jlinked:/app -w /here ubuntu:focal bash
root@200b4a98f9ee:/here# /app/bin/edurun 1 1
Result of sum is 2
 类似资料:
  • 问题内容: 我想对如何处理以下情况有清楚的了解: 我正在从activerecord模型中添加或删除属性,所以我想在生产中的ElasticSearch中更新其映射。 据我了解,我应该… 1-创建一个新索引并从mysql导入所有内容 这是正确的命令吗? 为了创建正确的映射,我应该已经在模型中更新了映射,对吗? 2-删除旧的映射并创建一个别名为 我会那样做,对吗? 3-重启应用 我是否缺少某些东西,或者

  • 问题内容: 是否可以将Oracle JDK随附的 tools.jar 与依赖于它的商业产品一起分发(即,根据Oracle许可条款,在法律上)?是或否,请提供参考以证实您的答案。 问题答案: 根据我的阅读,是的,但有一些警告: 您必须仅通过“ javac字节码编译器”(包括tools.jar;)重新分发整个Oracle JDK或整个JRE。允许重新分发与您的应用程序捆绑在一起。 重新分发的JDK或J

  • 最新版本的kafka支持精确一次语义(EoS)。为了支持这一概念,在每个消息中都添加了额外的详细信息。这意味着在你的消费者;如果打印消息的偏移量,它们不一定是连续的。这使得轮询一个主题以阅读最后提交的消息变得更加困难。 在我的例子中,consumer打印了如下所示的内容 问题:为了编写可重启的proudcer;我对话题进行了投票,并阅读了上一条消息的内容。在这种情况下;最后一条消息将是偏移量#5,

  • 我希望在进行计算时加快很多任务,但在写入表时合并成一组较小的分区。 下面给出了一个简单的演示示例,其中在执行过程中不支持重新分区。 我的预期输出是,映射操作发生在100个分区,最后收集发生在只有10个分区。 Spark似乎通过忽略重新分区优化了执行。如果有人能解释一下如何实现我的预期行为,那会很有帮助。

  • 最近引起我注意的是,当产生“新的最佳分数”时,一个步骤比其他步骤慢。这无疑是解决方案克隆,每一步都会产生“新的最佳分数”。 因此,如果“新的最佳分数”步骤不是连续的,这是可以的。例如,如果我们连续有50个步骤,解决方案克隆过程将执行50次。更聪明的方法是在序列结束时进行克隆过程(仅一次)。 这是一个可以很容易实现的东西,还是有其他东西可以阻止它? 另一个想法是在每个“新的最佳分数”步骤进行克隆,但

  • 错误:ExampleUtil模块找不到module-info.class 有没有一种方法可以使用不是模块的JAR来创建运行时映像?谢谢你。