Keywords: Groovy, Reflection, 反射
The Reflection of Groovy String constant style method.
Groovy支持以下的方法定义:
class A {
def "I am a method"() {
}
}
Groovy是继承Java的机制的,而Java显然是不支持这种函数定义命名的。然而实际上,你是用A.class.getMethods() 或 A.metaClass.getMethods() 都能获取到带空格的方法名,只是调用起来很麻烦,可能只能靠反射去invoke调用。所以一般这种写法主要用于TestSuite做单元测试或测试驱动开发。其实这个反射很简单,难的是下面的。
Spock是个很灵活的Java/Groovy的测试框架(Spock以后有空再介绍),基于JUnit。但是Spock中继承了Specfication的测试类的字符串常量方法,会在groovy compile的时候compile成一个名字类似$spock_feature_0_0 这样的名字,这时候想要找回原来的对应的常量就不是那么简单了。但既然JUnit能做到,我们就去看JUnit的run的源码,发现它会生成一个Sputnik的Runner,这是个Spock的类,其中它会有个private的SpecInfo的字段,里面包含了当前类的所有features。Spock把每个test method看作一个feature。
因此我只要new一个Sputnik,通过反射把SpecInfo中的信息拿出来,就能拿到按顺序所有排序的方法和对应编译后的方法名了了。
public static List<String> getAllSpecFeatures(String className) {
List<String> ret = new ArrayList<>();
Class klass = Class.forName(className);
Sputnik runner = new Sputnik(klass);
Method m = runner.class.getDeclaredMethod("getSpec");
m.setAccessible(true);
SpecInfo specInfo = m.invoke(runner);
List<FeatureInfo> featureInfos = specInfo.getAllFeatures();
for (FeatureInfo featureInfo : featureInfos) {
// original groovy method name
ret.add(featureInfo.getName()); // compiled method name, format like "$spock_feature_0_0"
println(featureInfo.featureMethod.reflection.name)
}
return ret;
}