我正在使用JavaCompiler
javax.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/*'";
但它们都没有奏效。有什么想法吗?
由于
注意:路径包含斜杠和反斜杠的原因是因为my程序在运行时识别环境并自动完成路径。
编辑:我使用的是tomcat 6和java 1.7.0_21
答案 0 :(得分:3)
通配符:由于使用java / javaw / javac时支持Java 1.6通配符,因此有更多信息:Windows / Solaris and Linux
示例:
javac -cp "lib/*" Test.java
这将lib目录中的所有.jar文件(不是.class!)用作类路径。这不应该与shell的*扩展相混淆。 -cp lib/*
扩展为-cp lib/a.jar lib/b.jar
,这是无效的参数语法。为避免这种情况,您必须添加引号:-cp "lib/*"
您的问题的原因:您正在尝试使用其Java API直接从源代码调用Java编译器。此源代码不包含通配符扩展。 JDK附带一个包装器二进制文件(javac,javadoc,javah,javap都是相同的二进制文件),它执行某些操作并最终调用编译器任务。此包装器还会扩展类路径中的通配符,因此编译器任务不再需要执行此操作(并且不会这样做)。请参阅Compiler Readme部分“build - &gt; Notes - &gt;启动器”。 Launcher sourcecode
<强>解决方案强>:
javac
。 (这是不推荐,因为它是一个简单问题的复杂且容易出错的解决方案)示例代码:
String classpath = buildClassPath("lib/", "test/", "lib/*");
System.out.println(classpath);
// output: lib/;test/;lib/a.jar;lib/b.jar;
此函数获取所有类路径条目并构建一个类路径。带有通配符的Classpath条目将被扩展。
/**
* 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();
}
答案 1 :(得分:1)
使用反斜杠或斜杠没有区别。但是你明显假设路径是自动全局的(就像普通的命令行一样)。这会发生 不 。因此,您可以像使用命令行arg
一样运行编译器-classpath 'C:\\tomcat6\\webapps\\myapp/WEB-INF/lib/*'
这不是你想要的。您可以查看java.io.File.listFiles(FileFilter)
甚至java.nio.file.Files.walkFileTree(Path, FileVisitor)
的API文档,以便更好地理解。
为了扩展一点,当你的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
而不是手动加入。