我最近遇到了一个问题。我正在编写一些单元测试,以执行我的库中负责代码生成的某些部分。代码接受configuration对象的输入,使用CodeModel编写代码,将这些类转储为字符串,然后使用Java6编译器api将它们动态编译成可用的类。
我遇到的障碍是我在生成的代码中测试的方法之一,它使用带有最终方法的类(Android Bundle),并在使用时抛出异常("java.lang.RuntimeException: Stub!")。因此,为了解决这个问题,我使用了PowerMock来模拟最终的方法。然而,在这种情况下,编译器API抛出一个NPE。我认为这是因为PowerMock在幕后使用的工具,但我不确定。
下面是一个例外:
at org.androidtransfuse.gen.classloader.MemoryFileManager.<init>(MemoryFileManager.java:11)
at org.androidtransfuse.gen.classloader.MemoryClassLoader.<init>(MemoryClassLoader.java:16)
at org.androidtransfuse.gen.ParcelableGeneratorTest.setup(ParcelableGeneratorTest.java:67)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at org.junit.internal.runners.MethodRoadie.runBefores(MethodRoadie.java:129)
at org.junit.internal.runners.MethodRoadie.runBeforesThenTestThenAfters(MethodRoadie.java:93)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.executeTest(PowerMockJUnit44RunnerDelegateImpl.java:296)MemoryFileManager.java:
class MemoryFileManager extends ForwardingJavaFileManager<JavaFileManager> {
public final Map<String, Output> map = new HashMap<String, Output>();
MemoryFileManager(JavaCompiler compiler) {
super(compiler.getStandardFileManager(null, null, null));
}
@Override
public Output getJavaFileForOutput
(Location location, String name, JavaFileObject.Kind kind, FileObject source) {
Output mc = new Output(name, kind);
this.map.put(name, mc);
return mc;
}
}MemoryClassLoader.java (使用MemoryFileManager):
public class MemoryClassLoader extends ClassLoader {
private final JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
private final MemoryFileManager manager = new MemoryFileManager(this.compiler);
public void add(String classname, String filecontent) {
add(Collections.singletonMap(classname, filecontent));
}
public void add(Map<String, String> map) {
List<Source> list = new ArrayList<Source>();
for (Map.Entry<String, String> entry : map.entrySet()) {
list.add(new Source(entry.getKey(), JavaFileObject.Kind.SOURCE, entry.getValue()));
}
this.compiler.getTask(null, this.manager, null, null, null, list).call();
}
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
synchronized (this.manager) {
Output mc = this.manager.map.remove(name);
if (mc != null) {
byte[] array = mc.toByteArray();
return defineClass(name, array, 0, array.length);
}
}
return super.findClass(name);
}
}一个非常基本的测试来演示这个问题:
@RunWith(PowerMockRunner.class)
@PrepareForTest(Parcel.class)
public class ExampleTest {
private static final String TEST_VALUE = "test value";
String classToCompile =
"import android.os.Parcel;\n" +
"\n" +
"public class ClassToCompile\n" +
"{\n" +
"\n" +
" private Parcel parcel;\n" +
"}";
@Test
public void test() throws ClassNotFoundException, IOException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, InstantiationException {
MemoryClassLoader classLoader = new MemoryClassLoader();
classLoader.add("ClassToCompile", classToCompile);
classLoader.loadClass("ClassToCompile");
}
}发布于 2012-05-24 22:38:13
Johan在Powermock Google Group上为我添加了这个answered,但我想我也应该在这里添加结果。
对我来说起作用的是从@RunWith(PowerMockRunner.class)方法到here描述的Java Agent方法的改变。
这需要我删除@RunWith注释,并将以下内容添加到我的Maven POM中:
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-surefire-plugin</artifactId>
<configuration>
<argLine>
-javaagent:${settings.localRepository}/org/powermock/powermock-module-javaagent/1.4.12/powermock-module-javaagent-1.4.12.jar
</argLine>
</configuration>
</plugin>在IDE中运行单元测试时,我必须添加
-javaagent: <jarpath>/powermock-module-javaagent-1.4.12.jar添加到运行配置中。
https://stackoverflow.com/questions/10695509
复制相似问题