当我们需要对一个对象中所有元素进行代码检查或者赋值检查等操作,或者想添加一些新的功能,但是当该对象对于产品已经很成熟,可能直接修改对象的类会产生新的缺陷。这时访问者可以帮助我们增添功能,而不影响原始类.
Visitor(访问者)模式:表示一个作用与对象结构中各个元素的操作。可以使你不改变原始类的情况下增加对这些元素的新操作。
java中字节码操纵技术ASM中对字节码的访问,修改就是通过Visitor模式实现的。下面以asm为例进行介绍。
定义如下:
该类主要是读取和解析读入的class字节码文件,accept接受指定观察者,进行相关操作。b是读取的class文件字节数组。字节码常见结构如下:
该抽象类定义如下:
该类定义了各种访问ClassReader中元素的操作,这些访问是按照一定顺序进行处理的。
定义如下:
public class ClassParser extends ClassVisitor {
public ClassParser(int i) {
super(i);
}
public ClassParser(ClassWriter cw) {
super(Opcodes.ASM5, cw);
}
@Override
public void visit(int i, int i1, String s, String s1, String s2, String[] strings) {
super.visit(i, i1+Opcodes.ACC_ABSTRACT, s, s1, s2, strings);
System.out.println("类名"+s);
}
@Override
public AnnotationVisitor visitAnnotation(String s, boolean b) {
System.out.println("注释:"+s + "可见性:" + b);
return super.visitAnnotation(s,b);
// return new AnnotationParserOne(262144);
}
@Override
public FieldVisitor visitField(int i, String s, String s1, String s2, Object o) {
System.out.println("字段:"+s + "描述符:"+ s1);
return super.visitField(i, s, s1, s2, o);
// if ("name".equals(s)) {
// return super.visitField(i, "str", s1, s2, o);
// }
// if ("age".equals(s)) {
// return super.visitField(i + Opcodes.ACC_FINAL, s, s1, s2, (Integer) 10);
// }
// return new FiledParser(262144);
}
}
主要是在访问该类时,输出类名、输出注释和可见性以及字段名称和对应描述符。
代码:
public class AsmReadClassInfo {
public static void main(String[] args) throws IOException {
ClassReader classReader = new ClassReader(AsmTestClass.class.getName());
ClassParser classParser = new ClassParser(262144);
classReader.accept(classParser, ClassReader.SKIP_CODE);
}
}
@Component
public class AsmTestClass {
@NotNull
private String name;
private static int age;
private static final Integer[] arr = {};
}
运行结果如下:
结果中字段描述符参考具体jvm字节码内容的定义。
当需要处理复杂对象元素,并希望在遍历时应用一些操作,可以使用访问者进行处理,同时不修改原对象类代码。
[1]. 《设计模式》