如何使用通配符将JavaCompiler的类路径设置为多个.jar文件
我正在使用javax.tools
JavaCompiler
编译一些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 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
通配符 :由于使用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都是相同的二进制文件),它执行某些操作并最终调用编译器任务。 此包装器还会扩展类路径中的通配符,因此编译器任务不再需要执行此操作(并且不会这样做)。 请参阅编译器自述文件部分“build – > Notes – > The launcher”。 启动器源代码 。
方案 :
- 一个非常糟糕的解决方案是通过Processbuilder调用
javac
。 ( 不建议这样做,因为对于一个简单的问题,它是一个复杂且容易出错的解决方案) - 自己展开通配符:
示例代码:
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(); }
使用反斜杠或斜杠没有区别。 但是你明显假设路径是自动全局的(就像普通的命令行一样)。 这不会发生。 因此,您可以像使用命令行arg一样运行编译器
-classpath 'C:\\tomcat6\\webapps\\myapp/WEB-INF/lib/*'
这不是你想要的。 您可以查看java.io.File.listFiles(FileFilter)
的API文档,甚至是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 options = Arrays.asList("-d", classesBaseDir, "-cp", sb.toString()) ...
从Java1.7开始,您还可以使用Files.newDirectoryStream(Path)
代替list(File)
。 使用1.8,你甚至可以调用join
而不是手动加入。
- 如何解决此错误引起:java.lang.ClassNotFoundException
- 如何使用JSP页面中的参数运行java类?
- jar文件外的application.properties如何
- 是否可以动态加载包中包含的所有属性文件? 即MyClass.class.getResource(’*。properties’);
- 在代码上获取ClassNotFoundException:“Class.forName(”com.microsoft.sqlserver.jdbc.SqlServerDriver“);”
- 在命令行运行java程序,我做错了什么?
- 控制servlet中的类路径
- Axis2客户端的最小类路径是什么?
- 让Rhino JS看Java类