2

我正在开发一个客户端-服务器 Java 应用程序,该应用程序可以在 Mac OS X 以及 Windows 和 Linux 上运行。该应用程序有几个不同的客户端模块,应该从启动器应用程序执行。我看过 ProcessBuilder 类,但我似乎不适合我。我在这里找到了一个线程 att stackoverflow 建议使用 Ant 的示例(here)。我实现了一个执行客户端模块的方法:

public void launchAnt(ApplicationData applicationData) {
    Project project = new Project();
    project.setBaseDir(new File(System.getProperty("user.dir")));
    project.init();
    DefaultLogger logger = new DefaultLogger();
    project.addBuildListener(logger);
    logger.setOutputPrintStream(System.out);
    logger.setErrorPrintStream(System.err);
    logger.setMessageOutputLevel(Project.MSG_INFO);
    System.setOut(new PrintStream(new DemuxOutputStream(project, false)));
    System.setErr(new PrintStream(new DemuxOutputStream(project, true)));
    project.fireBuildStarted();

    System.out.println("ApplicationLauncher.launch(): Running");
    Throwable caught = null;
    try {
        /**
         * Create Java task
         */
        Java javaTask = new Java();
        javaTask.setTaskName("Run " + applicationData.getApplication().name());
        javaTask.setProject(project);
        javaTask.setFork(false);
        javaTask.setFailonerror(true);
        javaTask.setClassname(applicationData.getClassName());

        /**
         * Working directory
         */
        File workdir = new File(System.getProperty("user.dir"));
        if(workdir.exists()) {
            javaTask.setDir(workdir);
        } else {
            System.out.println("ApplicationLauncher.launch(): ERROR: Unable to set workdir, " + workdir.getAbsolutePath() + " does note exist.");
        }

        /**
         * Classpath
         */
        Path path = new Path(project);
        Collection<String> classpaths = getClasspath();
        for (String classpath : classpaths) {
            Path currPath = new Path(project, new File(classpath).getAbsolutePath());
            path.add(currPath);
        }
        System.out.println("ApplicationLauncher.launch(): Classpath: " + path.toString());
        javaTask.setClasspath(path);

        /**
         * Arguments
         */
        Argument arg = javaTask.createArg();
        arg.setValue(applicationData.getArguments());

        /**
         * Initiate and execute
         */
        javaTask.init();
        int ret = javaTask.executeJava();
        System.out.println("ApplicationLauncher.launch(): Java task return code: " + ret);
    } catch (BuildException e) {
        caught = e;
    }

    project.log("ApplicationLauncher.launch(): Finished");
    project.fireBuildFinished(caught);
}

ApplicationData 是一个对象,其中包含有关要启动的应用程序的一些信息。此代码运行良好,应用程序已启动。问题出现在该行导致 ClassNotFoundException 的启动应用程序中:

UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());

这是堆栈跟踪:

java.lang.ClassNotFoundException: apple.laf.AquaLookAndFeel
    at org.apache.tools.ant.AntClassLoader.findClassInComponents(AntClassLoader.java:1361)
    at org.apache.tools.ant.AntClassLoader.findClass(AntClassLoader.java:1311)
    at org.apache.tools.ant.AntClassLoader.loadClass(AntClassLoader.java:1070)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:251)
    at java.lang.ClassLoader.loadClassInternal(ClassLoader.java:374)
    at java.lang.Class.forName0(Native Method)
    at java.lang.Class.forName(Class.java:242)
    at javax.swing.SwingUtilities.loadSystemClass(SwingUtilities.java:1788)
    at javax.swing.UIManager.setLookAndFeel(UIManager.java:484)

如果我尝试独立运行应用程序,它工作正常。启动器应用程序包含完全相同的失败行,并且在启动器中运行良好。启动应用程序的蚂蚁方式似乎有些不对劲。在从启动器执行应用程序时,我已经比较了 System.properties 和类路径,并且独立(有效)并且没有区别。

我只是不明白为什么找不到这个类。有没有其他人看到这个问题?任何建议表示赞赏!

谢谢!

4

1 回答 1

0

这些是一些最难调试的错误。代码在 Ant 之外运行良好是一个好兆头,这会留下类路径问题。检查项目的类路径,系统找不到“apple.laf.AquaLookAndFeel”。

于 2011-07-13T19:47:57.210 回答