3

我正在使用JavaCompilerofjavax.tools来编译一些 java 代码,并且我试图在我的文件中使用通配符classpath以包含所有.jar文件,但我失败了。

这是我的代码:

    String classpath = "C:\tomcat6\webapps\myapp/WEB-INF/lib/javax.ws.rs-api-2.0-m10.jar;"
+ "C:\\tomcat6\\webapps\\myapp/WEB-INF/lib/javax.persistence-2.1.0.jar";

    Iterable<String> options = Arrays.asList("-d", classesBaseDir,
                    "-classpath", classpath);

    JavaCompiler.CompilationTask task = compiler.getTask(null, fileManager,
                    diagnostics, options, null, file);
    boolean result = task.call();

上面的代码工作得很好。但是当我试图classpath改变

String classpath = "C:\\tomcat6\\webapps\\myapp/WEB-INF/lib/*";

它失败了

    compiler.err.doesnt.exist|package javax.ws.rs does not exist
...
    symbol:   class GET
      location: class com.my.oasis.resources.TestClass
    09/04/2014 14:27:09:030 | COMPILER_DIAGNOSTIC | compileResource() - compiler.err.cant.resolve.location|cannot find symbol
    ...

我还尝试了以下更改

String classpath = "\"C:\\tomcat6\\webapps\\myapp/WEB-INF/lib/*\"";
String classpath = "'C:\\tomcat6\\webapps\\myapp/WEB-INF/lib/*'";

但他们都没有工作。有任何想法吗?

谢谢

注意:路径包含斜杠和反斜杠的原因是因为我的程序在运行时识别环境并自动补全路径。

编辑:我正在使用 tomcat 6 和 java 1.7.0_21

4

2 回答 2

3

通配符:由于使用 java/javaw/javac 时支持 Java 1.6 通配符,更多信息:Windows / Solaris 和 Linux

例子:

javac -cp "lib/*" Test.java

这使用 lib 目录中的所有 .jar 文件(不是 .class!)作为类路径。这不应该与你的 shell 的 *-expansion 混淆。-cp lib/*扩展为-cp lib/a.jar lib/b.jar无效的参数语法。为了避免这种情况,您必须添加引号:-cp "lib/*"

问题的原因:您试图直接使用 Java API 从源代码调用 Java 编译器。此源代码不包含通配符扩展。JDK 附带了一个包装二进制文件(javac、javadoc、javah、javap 都是相同的二进制文件),它执行一些操作并最终调用编译器任务。这个包装器还扩展了类路径中的通配符,因此编译器任务不再需要这样做(现在也不需要了)。请参阅编译器自述文件部分“构建 -> 注释 -> 启动器”。启动器源代码

解决方案

  1. 一个非常糟糕的解决方案是javac通过 Processbuilder 调用。(不推荐这样做,因为它是一个简单问题的复杂且容易出错的解决方案)
  2. 自己展开通配符:

示例代码:

String classpath = buildClassPath("lib/", "test/", "lib/*");
System.out.println(classpath);
// output: lib/;test/;lib/a.jar;lib/b.jar;

此函数获取所有类路径条目并构建一个类路径。带有通配符的类路径条目将被扩展。

/**
 * This function builds a classpath from the passed Strings
 * 
 * @param paths classpath elements
 * @return returns the complete classpath with wildcards expanded
 */
private static String buildClassPath(String... paths) {
    StringBuilder sb = new StringBuilder();
    for (String path : paths) {
        if (path.endsWith("*")) {
            path = path.substring(0, path.length() - 1);
            File pathFile = new File(path);
            for (File file : pathFile.listFiles()) {
                if (file.isFile() && file.getName().endsWith(".jar")) {
                    sb.append(path);
                    sb.append(file.getName());
                    sb.append(System.getProperty("path.separator"));
                }
            }
        } else {
            sb.append(path);
            sb.append(System.getProperty("path.separator"));
        }
    }
    return sb.toString();
}
于 2014-04-10T13:08:59.330 回答
1

使用反斜杠或斜杠没有区别。但是您显然假设路径是自动通配的(就像普通命令行一样)。这不会发生。因此,您可以像使用命令行参数一样运行编译器

-classpath 'C:\\tomcat6\\webapps\\myapp/WEB-INF/lib/*'

这不是你想要的。您可以查看 API 文档,java.io.File.listFiles(FileFilter)甚至java.nio.file.Files.walkFileTree(Path, FileVisitor)获得更好的理解。

为了对此进行扩展,当您的 shell 看到 C:\tomcat6\webapps\myapp/WEB-INF/lib/* 时,它会将其扩展为以空格分隔的目录中任何内容的列表WEB-INF/lib。这称为通配,或者因为它是自动完成的,所以称为自动通配。

现在 Java 不这样做,但您可以用几行代码自己构建它:

StringBuilder sb = new StringBuilder();
String[] filenames = new File("C:\\tomcat6\\webapps\\myapp/WEB-INF/lib/").list();
for(int i = 0; i < filenames.length; i++) {
    if(i > 0) sb.append(File.pathSeparatorChar); // separate with ':' or ';' on Win
    sb.append(filenames[i]); // append the filename
}
Iterable<String> options = Arrays.asList("-d", classesBaseDir, "-cp", sb.toString())
...

从 Java1.7 开始,您也可以Files.newDirectoryStream(Path)使用list(File). 使用 1.8,您甚至可以调用join而不是手动加入。

于 2014-04-09T15:20:01.857 回答