Java - 在不同的进程中启动另一个类的主
Posted
技术标签:
【中文标题】Java - 在不同的进程中启动另一个类的主【英文标题】:Java - start another class' main in a different process 【发布时间】:2014-02-07 06:10:51 【问题描述】:我需要一种干净的方式来启动带有 GUI 的 Java 程序的许多实例,并且我想以编程方式完成它。我要运行的“程序”只是一个 .class 文件(带有 main 方法的已编译 .java 文件),它应该显示一个 GUI 并独立于其他运行(作为它自己的进程)。我还需要向该程序传递一些参数。
检查 EDIT5 以获取完整的工作解决方案代码。
这是应该启动许多进程的类
package startothermain;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
public class Starter
public static void main(String[] args)
int starts = 4;
for (int i = 0; i < starts; ++i)
System.out.println("Starting an app");
ProcessBuilder pb = new ProcessBuilder("java.exe", "-cp", "bin", "Started", "arg0");
try
pb.start();
catch (IOException ex)
Logger.getLogger(Starter.class.getName()).log(Level.SEVERE, null, ex);
这是应该启动并显示 GUI 的类
package startothermain;
import javax.swing.JOptionPane;
public class Started
public static void main(String[] args)
JOptionPane.showMessageDialog(null, args[0]);
我尝试了在其他答案中找到的 ProcessBuilder 和 Runtime.getRuntime() 建议,但它们似乎不起作用。我总是收到某种“未找到”错误,比如这个
SEVERE: null
java.io.IOException: Cannot run program "java.exe": error=2, No such file or directory
at java.lang.ProcessBuilder.start(ProcessBuilder.java:1041)
at startothermain.Starter.main(Starter.java:35)
Caused by: java.io.IOException: error=2, No such file or directory
at java.lang.UNIXProcess.forkAndExec(Native Method)
at java.lang.UNIXProcess.<init>(UNIXProcess.java:135)
我正在使用 NetBeans 并单击“运行”按钮,因此 .java 文件应该在同一个文件夹中正确编译。我希望这不是 IDE 创建的 src/build 文件夹的问题。
编辑:我试图找到 java.exe,但我在 Linux 上。叹。我将它更改为“java”,但我仍然遇到同样的问题。应该设置路径变量。此外,我担心,如果我给它一个完整的路径,它会变得不可移植。
尝试在 Linux 上获取 JAVA_HOME
System.getenv("JAVA_HOME"); // returns null
System.getProperty("java.home"); // returns /usr/lib/jvm/java-7-openjdk-amd64/jre
在 Windows 上
System.getenv("JAVA_HOME"); // returns C:\Program Files\Java\jdk1.7.0_51
System.getProperty("java.home"); // returns C:\Program Files\Java\jdk1.7.0_51\jre
EDIT2:这个新代码不会产生错误,但也不会打开任何 GUI。我在 Windows 和 Linux 上都试过了,结果相同。
String javaHome = System.getProperty("java.home");
ProcessBuilder pb = new ProcessBuilder(javaHome + "/bin/java", "-cp", "bin", "Started", "arg0");
EDIT3:我重定向了 processbuilder(不是创建的进程)的错误和输出流,结果 Java ClassLoader 找不到该类。我猜问题不是JAVA_HOME,而是路径问题。
这是新代码
package startothermain;
import java.io.File;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
public class Starter
public static void main(String[] args)
int starts = 4;
System.out.println(System.getProperty("java.home")); // prints C:\Program Files\Java\jdk1.7.0_51\jre
System.out.println(System.getenv("JAVA_HOME")); // prints C:\Program Files\Java\jdk1.7.0_51
System.out.println("Working Directory = "
+ System.getProperty("user.dir")); // prints C:\Users\Agostino\Documents\NetBeansProjects\StartOtherMain
for (int i = 0; i < starts; ++i)
System.out.println("Starting an app");
ProcessBuilder pb = new ProcessBuilder("java", "build.classes.startothermain.Started");
File log = new File("log");
pb.redirectOutput(log);
pb.redirectError(log);
try
pb.start();
catch (IOException ex)
Logger.getLogger(Starter.class.getName()).log(Level.SEVERE, null, ex);
在日志文件中我发现了这个错误
java.lang.NoClassDefFoundError: build/classes/startothermain/Started (wrong name: startothermain/Started)
at java.lang.ClassLoader.defineClass1(Native Method)
at java.lang.ClassLoader.defineClass(ClassLoader.java:800)
at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:142)
at java.net.URLClassLoader.defineClass(URLClassLoader.java:449)
at java.net.URLClassLoader.access$100(URLClassLoader.java:71)
at java.net.URLClassLoader$1.run(URLClassLoader.java:361)
at java.net.URLClassLoader$1.run(URLClassLoader.java:355)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(URLClassLoader.java:354)
at java.lang.ClassLoader.loadClass(ClassLoader.java:425)
at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
at java.lang.ClassLoader.loadClass(ClassLoader.java:358)
at sun.launcher.LauncherHelper.checkAndLoadMain(LauncherHelper.java:482)
Exception in thread "main"
EDIT4:现在代码可以工作了,但是如果我尝试在 Started 类中使用 .jar 库,则找不到它。
查看这个使用 .jar 库 (JavaTuples) 的 Started 类,该库通过 NetBeans 添加到项目库中
package startothermain;
import java.io.File;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
public class Starter
public static void main(String[] args)
int starts = 1;
for (int i = 0; i < starts; ++i)
System.out.println("Starting an app");
ProcessBuilder pb = new ProcessBuilder();
String fullClassName = Started.class.getName();
pb.command("java", "-cp", "./build/classes", fullClassName, "myArg");
File log = new File("log");
pb.redirectOutput(log);
pb.redirectError(log);
try
pb.start();
catch (IOException ex)
Logger.getLogger(Starter.class.getName()).log(Level.SEVERE, null, ex);
固定的入门类
package startothermain;
import java.io.File;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
public class Starter
public static void main(String[] args)
int starts = 1;
for (int i = 0; i < starts; ++i)
System.out.println("Starting an app");
ProcessBuilder pb = new ProcessBuilder();
String fullClassName = Started.class.getName();
pb.command("java", "-cp", "./build/classes", fullClassName, "myArg");
File log = new File("log");
pb.redirectOutput(log);
pb.redirectError(log);
try
pb.start();
catch (IOException ex)
Logger.getLogger(Starter.class.getName()).log(Level.SEVERE, null, ex);
ProcessBuilder 日志中的错误
Exception in thread "main" java.lang.NoClassDefFoundError: org/javatuples/Pair
at startothermain.Started.main(Started.java:28)
Caused by: java.lang.ClassNotFoundException: org.javatuples.Pair
at java.net.URLClassLoader$1.run(URLClassLoader.java:366)
at java.net.URLClassLoader$1.run(URLClassLoader.java:355)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(URLClassLoader.java:354)
at java.lang.ClassLoader.loadClass(ClassLoader.java:425)
at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
at java.lang.ClassLoader.loadClass(ClassLoader.java:358)
... 1 more
EDIT5:工作代码
入门类
package startothermain;
import java.io.File;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
public class Starter
public static void main(String[] args)
int starts = 1;
for (int i = 0; i < starts; ++i)
System.out.println("Starting an app");
ProcessBuilder pb = new ProcessBuilder();
String fullClassName = Started.class.getName();
String pathToClassFiles = new File("./build/classes").getPath();
String pathSeparator = File.pathSeparator; // ":" on Linux, ";" on Windows
String pathToLib = new File("./lib/javatuples-1.2.jar").getPath();
pb.command("java", "-cp", pathToLib + pathSeparator + pathToClassFiles, fullClassName, "myArg");
File log = new File("log" + i + ".txt"); //debug log for started process
try
pb.redirectOutput(log);
pb.redirectError(log);
pb.start();
catch (IOException ex)
Logger.getLogger(Starter.class.getName()).log(Level.SEVERE, null, ex);
开始上课
package startothermain;
import javax.swing.JOptionPane;
import org.javatuples.Pair;
public class Started
public static void main(String[] args)
Pair<String, Integer> pair = Pair.with("One", 1);
JOptionPane.showMessageDialog(null, args[0] + " " + pair);
【问题讨论】:
您的堆栈跟踪似乎表明它找不到 java.exe,请尝试指定绝对路径。 或者将 java.exe 添加到您的系统文件路径中。您经常使用的任何程序都可以放在里面。 当您运行某种 *nix 或 Mac OS 时,如果/usr/bin/java", "-cp", ...
不适合您,请尝试使用类似 "/bin/bash", "-c", "/usr/bin/java -cp ..."
的命令。初级调用将在您的终端中执行 java 命令
@user1600770 你可以查找JAVA_HOME
environment variable 然后调用/bin
目录中的java 可执行文件。取决于您正在运行的OS
@RomanVottner 我用我的尝试更新了这个问题
【参考方案1】:
根据您的要求,我在回答中总结了我的 cmets。
您的第一个问题是您尝试在 Linux 或 Mac 机器上调用 java.exe
,这不符合 Microsoft 将文件类型包含在名称中的约定。因此 Linux 和 Mac 通常使用java
代替。
可执行文件的路径也因计算机而异。因此,可能最通用的方法是使用JAVA_HOME
环境变量来定位java 可执行文件,该环境变量可以通过System.getenv("JAVA_HOME");
或System.getProperty("java.home");
在Java 中获取,但是请注意,它们可能不会明显指向所需的目录。尤其是某些 Linux 发行版将所有二进制文件放在 /bin
或 /usr/bin
中,因此 $JAVA_HOME/bin/java
可能不可用。在这种情况下,您应该创建一个指向可执行文件的(硬)链接。
如果未设置路径,您可以在您正在执行应用程序的控制台会话中手动设置它(在 Windows 上 set JAVA_HOME=C:\Program Files\Java
(或在较新的 Windows 版本上为 setx)在 Linux export JAVA_HOME=/opt/java
上)。这也可以通过user-priviledges 完成
在处理进程时,进一步强烈建议take care of in- and output-streams。链接的文章深入解释了为什么您应该注意它 - 不仅仅是捕获被调用进程抛出的异常。
在调用进程时(尤其是使用 Java 可执行文件),您有几个选项:
您可以使用 new ProcessBuilder("java", "-cp", "path/to/your/binaries", "package.ClassName");
直接调用 Java 进程(假设 Java 在您的 PATH 上),也可以通过 shell 调用 Java 可执行文件 - 在 Linux f.e. 上。你也可以使用new ProcessBuilder("/bin/bash", "-c", "java -cp path/to/your/binaries package.ClassName");
(尽管引物显然更可取)。
在动态加载类/库时,您必须使用完全限定的类名。因此,如果您在项目的根目录中调用 Java 进程,并且您的构建工具在 ./build/classes
中生成类文件,并且您的类 Test
位于包 testpackage
中,那么您最终会得到以下设置:./build/classes/testpackage/Test.class
。要启动一个调用 Test.class
中包含的 main 方法的新 Java 进程,您必须使用以下命令:java -cp ./build/classes testpackage.Test
否则 Java 将无法找到 Test
的类定义并执行 main-method。
必须将缺少的依赖项添加到调用 Java 命令的类路径 (-cp ...
) 段中。 F.e:java -cp lib/jar1.jar;lib/jar2.jar;build/classes/* package.ClassName
。多个档案或目录由;
分隔,星号*
也可以包含目录中的所有内容。
还有一点需要注意:如果您曾经尝试发布您的“应用程序”,则需要将此代码调整为更通用的版本(属性文件 fe),因为路径可能完全不同,这很可能会失败。
如果我忘记了什么,请告诉我。
【讨论】:
+1 谢谢。这行得通,但我刚刚发现了另一个(侧面)问题。检查 EDIT4。当我从外部启动该过程时,它找不到一个简单的库。我应该开始另一个问题吗? 您需要在您调用的Java 命令的-cp ...
段中提供包含所需类的jar。示例:java -cp lib/jar1.jar;lib/jar2.jar;build/classes package.ClassName
或简单的 java -cp lib/*;build/classes package.ClassName
【参考方案2】:
您是否尝试从命令提示符运行 java.exe,如果它在那里不起作用,您需要设置 JAVA 安装的 java 路径这可以通过在系统变量中设置变量 JAVA_PATH 来完成,这应该指向您的jdk bin 文件夹。
如果这不起作用,那么我认为您需要提供 JAVA.exe 的完整路径,因为程序正在尝试查找此文件并且它无法找到该文件并给出错误
相反,您可以尝试创建一个更好的解决方案的线程,因为线程使用的资源更少且效率更高
【讨论】:
我猜 JAVA_PATH 已设置,当我在终端中运行“which java”时,我得到“/usr/bin/java”。制作线程不是一个选项,因为我想启动完全独立的程序,然后关闭 Starter。如果存在不需要调用 java.exe 的建议,我愿意接受。顺便说一句,我在 Linux 上,为什么我要调用 exe?! 好的,我遇到了您的问题,因此您的程序正在尝试查找它无法找到的 java.exe。作为一个文件,所以你需要给 java.util 提供绝对路径。此外,如果您想关闭启动应用程序,请使用 JAVAW,因为当您尝试执行程序时它不会打开黑色终端窗口。 @user1600770 你在运行什么系统,java 位于/usr/bin/java
所以无论是 *nix 还是 Mac,你尝试调用 java.exe
- 这是一个 Windows 可执行文件
我尝试了“javaw”而不是“java.exe”,但我仍然得到“未找到错误”。有没有办法以便携的方式做到这一点?
是的@Roman Vottner 是正确的,您需要为您的平台指定 java 可执行文件的完整路径以上是关于Java - 在不同的进程中启动另一个类的主的主要内容,如果未能解决你的问题,请参考以下文章