本文共 1394 字,大约阅读时间需要 4 分钟。
原意是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; }