我有一个带有几个模块的maven项目,即
<module>backend</module> <!-- provides annotations -->
<module>annotationProcessor</module> <!-- processes ann., generates files -->
<module>mainprog</module> <!-- uses annotations/files -->
backend
为注释类提供了注释类MyAnnotation
。
mainprog
包含 Mainprog.java
,它定义了一个带有@MyAnnotation
注释的类。在运行时,此类尝试通过getResourceAsStream("Mainprog.properties")
(尚不存在)加载文件。
annotationProcessor
有一个类MyAnnotationProcessor
maven 执行并找到我的注释。
处理器应根据注释处理器收集的信息创建文件Mainprog.properties
。
在执行/测试Mainprog时,我无法将属性文件放在找到它的位置。
package demo;
@MyAnnotation
public class Mainprog {
}
目前我在测试类中执行此操作,但稍后这将在类本身中。
package demo;
import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;
import org.junit.Test;
public class MainprogTest {
Class testclass = Mainprog.class;
@Test
public void testPropertiesFile() throws IOException {
String fn = testclass.getCanonicalName().replace('.', '/') + ".properties";
System.err.println("loading: '"+fn+"'");
InputStream in = getClass().getResourceAsStream(fn);
Properties prop = new Properties();
prop.load(in);
in.close();
}
}
目前这样运行:
loading: 'demo/Mainprog.properties'
Tests in error:
testPropertiesFile(demo.MainprogTest)
使用NullPointerException
,因为流返回null
,即不存在。
尽管文件存在(但它在正确的位置?):
towi@havaloc:~/git/project/mainprog$ find . -name Mainprog.properties
./src/java/demo/Mainprog.properties
./target/classes/demo/Mainprog.properties
package demo;
import com.github.javaparser.*;
import com.github.javaparser.ast.*;
import javax.annotation.processing.*;
import javax.lang.model.element.*;
@SupportedAnnotationTypes({"demo.MyAnnotation"})
public class MyAnnotationProcessor extends AbstractProcessor {
@Override
public boolean process(Set<? extends TypeElement> elements, RoundEnvironment env) {
for (TypeElement te : elements) {
for (Element e : env.getElementsAnnotatedWith(te))
{
processAnnotation(e);
}
}
return true;
}
private void processAnnotation(Element elem) {
final TypeElement classElem = (TypeElement) elem;
...
final String prefix = System.getProperty("user.dir").endsWith("/"+"mainprog") ? "." : "mainprog";
final String className = classElem.getQualifiedName().toString();
String fileName = prefix + "/src/java/" + className.replace('.', '/') + ".java";
FileInputStream in = new FileInputStream(fileName);
final CompilationUnit cu = JavaParser.parse(in);
final CallGraph graph = ...
generateInfoProperties(classElem, fileName, graph);
}
private void generateInfoProperties(TypeElement classElem, String inFilename, CallGraph graph) throws IOException {
final File outFile = new File(inFilename
.replace("/src/java/", "/src/java/") // <<< WHERE TO ???
.replace(".java", ".properties"));
outFile.getParentFile().mkdirs();
try (PrintWriter writer = new PrintWriter(outFile, "UTF-8")) {
final Properties ps = new Properties();
graph.storeAsProperties(ps);
ps.store(writer, inFilename);
writer.close();
}
}
}
正如你所看到的,有很多猜测和&#34;启发式&#34;处理目录名时继续。所有System.getProperty("user.dir")
和replace("/src/java/", "/src/java/")
可能都是错误的,但更好的是什么?
在Maven,我有4个poms,当然
pom.xml
backend/pom.xml
annotationProcessor/pom.xml
mainprog/pom.xml
在我看来,只有一个包含任何注释,即在mainprog/pom.xml
中执行注释处理器:
<project>
....
<dependencies>
<dependency>
<groupId>project</groupId>
<artifactId>backend</artifactId>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>project</groupId>
<artifactId>annotationProcessor</artifactId>
<scope>compile</scope>
</dependency>
</dependencies>
<build>
<finalName>mainprog</finalName>
<sourceDirectory>src/java</sourceDirectory>
<resources>
<resource>
<directory>${basedir}/src/conf</directory>
<targetPath>META-INF</targetPath>
</resource>
<resource>
<directory>${basedir}/web</directory>
</resource>
<resource>
<directory>${basedir}/src/java</directory>
<includes>
<include>**/*.xml</include>
<include>**/*.properties</include>
<include>**/*.wsdl</include>
<include>**/*.xsd</include>
</includes>
</resource>
</resources>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<configuration>
<annotationProcessors>
<annotationProcessor>demo.MyAnnotationProcessor
</annotationProcessor>
</annotationProcessors>
</configuration>
</plugin>
...
</plugins>
</build>
</project>
我想通过将文件生成到/src/java/
然后让<resource><directory>${basedir}/src/java
和<include>**/*.properties
就足够了,但似乎并非如此。那是为什么?
答案 0 :(得分:2)
使用提供的Filer
,可以使用processingEnv.getFiler()
获取。如果使用它创建源文件,编译器将在下一轮编译它,您不必担心配置Maven来编译生成的源文件。