原意是spring配置文件的propertylist可以一一映射成一个枚举类,不过spring依赖注入也需要先定义好一个枚举类。
本想spring有的property,都能优雅的使用它,如下图:
不能依赖注入成枚举,构造一个枚举总行吧,结果枚举是一个单例的私有构造函数的类型。
那想想用反射用ASM动态添加对象属性总可以吧,也比较优雅,如下图:
想法是美好的,现实是残酷的,ASM的访问者模式并不适合预编译,想要达到“动态枚举”般的优雅确实没有找到好的方法。
ASM比较适合偷改class内容、函数体,ASM有个坑就是,字节码修改后,想要再获得对象实例就不可能了,报错如下图:
不知道“动态枚举”、“枚举泛型”这些坑什么时候能填上?
public static Bean getEnum(){
String filterName = "partyCountFilter";
if(filterName==null){
return instance = new Bean();
}
ApplicationContext context=new ClassPathXmlApplicationContext("classpath:/spring/spring-config-filter.xml");
AbstractRuleFilter bean=context.getBean(filterName, AbstractRuleFilter.class);
List<String> dataList = bean.getRuleDataName();
if(dataList==null || dataList.size()==0){
return instance = new Bean();
}
Class filterClass = Bean.class;
try{
ClassReader cr = new ClassReader(filterClass.getCanonicalName());
ClassWriter cw = new ClassWriter(cr,false);
ClassAdapter ca = null;
for(String dataName:dataList){
ca = new AddFieldAdapter(cw, Opcodes.ACC_PUBLIC,dataName, Type.getDescriptor(String.class),dataName);
cr.accept(ca,false);
}
byte[] bys = cw.toByteArray();
MyClassLoader classLoader = new MyClassLoader();
Class clazz = classLoader.defineClassFromClassFile(
"*.asm.Bean", bys);
try {//利用反射方式,访问
instance = (Bean)clazz.newInstance();
} catch (Exception e) {
e.printStackTrace();
}
}catch(Exception e){
e.printStackTrace();
}
return instance;
}
本文探讨了尝试使用ASM库动态地为枚举类添加属性以及实现类似功能的过程。作者发现在Spring框架中,无法直接将property依赖注入为枚举类型,并发现ASM虽然适合修改类内容和函数体,但不适用于预编译时创建动态枚举。在尝试过程中遇到了ASM修改字节码后无法实例化对象的问题,提出了对“动态枚举”和“枚举泛型”解决方案的期待。
1298

被折叠的 条评论
为什么被折叠?



