gpt4 book ai didi

java - 复制jar中的 `export`命令来设置环境变量

转载 作者:行者123 更新时间:2023-12-02 09:59:56 26 4
gpt4 key购买 nike

场景

在使用ProcessBuilder执行一系列命令时,我注意到我目前无法设置环境变量,使其在执行一组命令后保持“已知”状态。

问题

如何在 .jar 文件中重新创建 export TASKDDATA=/var/taskd 命令的效果*?

尝试 0

ProcessBuilder environment variable in java提供了一种为每个特定命令设置环境变量的方法,但是当我执行该解决方案的 .jar 并检查执行后环境变量 $u 是否仍然设置,我发现不是。而 $TASKDDATA 在执行后仍然保持设置状态。举例说明:

a@DESKTOP-desktopName:/mnt/e$ echo $TASKDDATA

a@DESKTOP-desktopName:/mnt/e$ TASKDDATA=/var/taskd
a@DESKTOP-desktopName:/mnt/e$ echo $TASKDDATA
/var/taskd
a@DESKTOP-desktopName:/mnt/e$ sudo java -jar autoInstallTaskwarrior.jar
[sudo] password for a:
Process ended with rc=0

Standard Output:

util/


Standard Error:


a@DESKTOP-desktopName:/mnt/e$ echo $TASKDDATA
/var/taskd
a@DESKTOP-desktopName:/mnt/e$ echo $u

尝试 1

对于单个命令,环境变量可以使用我在 Java ProcessBuilder how to get binary output from command 中编写的解决方案。但是,这不会为需要再次设置的第二个命令保留任务变量。但使用export命令时,不需要再次设置环境变量。具体来说,当 java 代码完成并且用户想要输入需要环境变量的任何其他命令时,这种差异就会显现出来。在这种情况下,用户需要先再次键入导出命令。

尝试 2

当使用 sudo -s 打开新 shell 以获取 root 权限时,会出现另一个差异。在 .jar 文件中设置环境不仅需要为每个单独的命令再次设置,而且环境变量不会传递到具有 root 权限的新 shell。例如执行以下命令:

commandLines[53] = new String[4];
commandLines[53][0] = "sudo";
commandLines[53][1] = "-s";
commandLines[53][2] = "taskdctl";
commandLines[53][3] = "start";
commands[53].setCommandLines(commandLines[53]);
commands[53].setEnvVarContent("/var/taskd");
commands[53].setEnvVarName("TASKDDATA");
commands[53].setWorkingPath("/usr/share/taskd/pki");

commandLines[54] = new String[5];
commandLines[54][0] = "sudo";
commandLines[54][1] = "-s";
commandLines[54][2] = "task";
commandLines[54][3] = "sync";
commandLines[54][4] = "init";
commands[54].setCommandLines(commandLines[54]);
commands[54].setEnvVarContent("/var/taskd");
commands[54].setEnvVarName("TASKDDATA");
commands[54].setWorkingPath("/usr/share/taskd/pki");

返回:

53RUNNINGCOMMAND=sudo -s taskdctl start
The TASKDDATA variable must be set.

54RUNNINGCOMMAND=sudo -s task sync
Could not connect to 0.0.0.0 53589
Sync failed. Could not connect to the Taskserver.
Syncing with 0.0.0.0:53589

注1

在执行 .jar 之前设置环境变量 $TASKDDATA=/var/taskd: TASKDDATA=/var/taskd sudo java -jar autoInstallTask​​warrior。 jar 不能确保环境变量 $TASKDDATA 在执行 .jar 文件后仍然可用。此外,它超出了问题的范围,因为它不是在 .jar 文件内设置的,而是在 .jar 文件外部设置的。

注2

我知道使用 export 命令作为由 processbuilder 执行的命令是不合适的,就像 cd 命令一样。

*这就是为什么问题集中在重现设置环境变量的“长期/持久”效果/可用性,而不是“如何执行导出命令”。

最佳答案

when I execute the .jar of that solution and check if the environment variable $u is still set after execution, I find it is not.

这是预期的行为。一些操作系统支持全局“环境”变量的概念。但不是 UNIX。在类似 UNIX 的操作系统中,每个进程都有自己的环境变量的私有(private)副本。一个进程不能修改另一个进程的环境。这也是为什么更改进程中的当前工作目录不会更改其父进程的 cwd。每个进程都有自己的cwd。

解决该限制的通常方法是让子进程将 var=val 对写入 stdout,然后父 shell 评估该输出以在其环境中设置变量。为了便于说明,假设该命令是以下 shell 脚本,名为 myscript.sh,而不是 Java 程序:

#!/bin/sh
echo VAR_A=val_a
echo VAR_B=val_b

然后父 shell 就会这样做

export $(./myscript.sh)

关于java - 复制jar中的 `export`命令来设置环境变量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55709684/

26 4 0
文章推荐: c++ - 无法推断函数返回类型
文章推荐: JAVA 将 AES key 、初始化 vector 和数据从加密文件中分离出来,并在加密过程中使用这 3 种组合
文章推荐: java - 如何使用 JPA native 查询将 List 转换为 POJO