Asm是很好的ByteCode generator 和 ByteCode reader。Asm提供了ClassVisitor来访问Class中的每个元素。当用ClassReader来读取Class的字节码时,每read一个元素,ASM会调用指定的ClassVisitor来访问这个元素。这就是访问者模式。利用这个特点,当ClassVisitor访问Class的Annotation元素时,我们会把annotation的信息记录下来。这样就可以在将来使用这个Annotation。
举一个例子吧,我们用AnnotationFaker来标注一个Class,也就是Annotation的target是Type级别。当我们发现某个class是用AnnotationFaker标注的,我们就load这个class到jvm中,并初始化,否则免谈。
1.AnnotationFaker: annnotation用来标注需要初始化的class
1
package com.oocl.isdc.sha.frm.test.config;
2data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
3
import java.lang.annotation.ElementType;
4
import java.lang.annotation.Retention;
5
import java.lang.annotation.RetentionPolicy;
6
import java.lang.annotation.Target;
7data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
8
@Retention(RetentionPolicy.RUNTIME)
9data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
@Target(
{ElementType.TYPE})
10data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public @interface AnnotationFaker
{
11
}
12data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
2.ClassFaker: 被AnnotationFaker标注的class
1
package com.oocl.isdc.sha.frm.test.config;
2data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
3
@AnnotationFaker
4data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public class ClassFaker
{
5data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void hello()
{
6
System.out.println("hello world, load me success!");
7
}
8
}
9data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
3.
ClassVisitorFaker:ClassVisitor的一个实现,用来得到class上的Annotation
1
package com.oocl.isdc.sha.frm.test.config;
2data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
3
import java.util.ArrayList;
4
import java.util.List;
5data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
6
import org.objectweb.asm.AnnotationVisitor;
7
import org.objectweb.asm.Attribute;
8
import org.objectweb.asm.ClassVisitor;
9
import org.objectweb.asm.FieldVisitor;
10
import org.objectweb.asm.MethodVisitor;
11
import org.objectweb.asm.tree.AnnotationNode;
12data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
13data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public class ClassVisitorFaker implements ClassVisitor
{
14data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
15
public List<AnnotationNode> visibleAnnotations;
16data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
17
public List<AnnotationNode> invisibleAnnotations;
18
19data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void visit(int version, int access, String name, String signature, String superName, String[] interfaces)
{
20
21
}
22data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
23data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public AnnotationVisitor visitAnnotation(String desc, boolean visible)
{
24
AnnotationNode an = new AnnotationNode(desc);
25data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (visible)
{
26data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (visibleAnnotations == null)
{
27
visibleAnnotations = new ArrayList<AnnotationNode> (1);
28
}
29
visibleAnnotations.add(an);
30data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else
{
31data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (invisibleAnnotations == null)
{
32
invisibleAnnotations = new ArrayList<AnnotationNode> (1);
33
}
34
invisibleAnnotations.add(an);
35
}
36
return an;
37
}
38data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
39data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void visitAttribute(Attribute attr)
{
40
}
41data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
42data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void visitEnd()
{
43
}
44data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
45data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public FieldVisitor visitField(int access, String name, String desc, String signature, Object value)
{
46
return null;
47
}
48data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
49data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void visitInnerClass(String name, String outerName, String innerName, int access)
{
50
}
51data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
52data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions)
{
53
return null;
54
}
55data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
56data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void visitOuterClass(String owner, String name, String desc)
{
57
}
58data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
59data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void visitSource(String source, String debug)
{
60
}
61data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
62data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public List<AnnotationNode> getVisibleAnnotations()
{
63
return visibleAnnotations;
64
}
65data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
66data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public List<AnnotationNode> getInvisibleAnnotations()
{
67
return invisibleAnnotations;
68
}
69
}
70data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
4.
ClassParser: main class, 分析classpath上的class,如果是用AnnotationFaker标注的class,我们就初始化它。
1
package com.oocl.isdc.sha.frm.test.config;
2data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
3
import java.io.File;
4
import java.io.IOException;
5
import java.net.URISyntaxException;
6
import java.net.URL;
7
import java.net.URLClassLoader;
8
import java.net.URLDecoder;
9
import java.util.Enumeration;
10
import java.util.HashSet;
11
import java.util.Iterator;
12
import java.util.List;
13
import java.util.Set;
14
import java.util.zip.ZipEntry;
15
import java.util.zip.ZipException;
16
import java.util.zip.ZipFile;
17data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
18
import org.objectweb.asm.ClassReader;
19
import org.objectweb.asm.Type;
20
import org.objectweb.asm.tree.AnnotationNode;
21data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
22data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public class ClassParser
{
23
@SuppressWarnings("unchecked")
24
public static void main(String[] args) throws IOException, ClassNotFoundException, InstantiationException,
25data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
IllegalAccessException, URISyntaxException
{
26
Set<String> result = new HashSet<String>();
27data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
28data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (getClassLoader() instanceof URLClassLoader)
{
29
URL[] urls = ((URLClassLoader) getClassLoader()).getURLs();
30data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
for (URL u : urls)
{
31
File file = new File(u.toURI());
32data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (file.isDirectory())
{
33
handleDirectory(result, file, null);
34data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else if (file.getName().toLowerCase().endsWith(".jar"))
{
35
handleArchive(result, file);
36
}
37
}
38data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
}else
{
39
Enumeration<URL> urls = getClassLoader().getResources(".");
40data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
while (urls.hasMoreElements())
{
41
String urlPath = urls.nextElement().getFile();
42
// System.out.println(urlPath);
43
urlPath = URLDecoder.decode(urlPath, "UTF-8");
44data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (urlPath.startsWith("file:"))
{
45
urlPath = urlPath.substring(5);
46
}
47data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (urlPath.indexOf('!') > 0)
{
48
urlPath = urlPath.substring(0, urlPath.indexOf('!'));
49
}
50data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
51
final File file = new File(urlPath);
52data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (file.isDirectory())
{
53
handleDirectory(result, file, null);
54
}
55data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
56
}
57
}
58data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
59
60data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
61data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
for (String clsRsName : result)
{
62
ClassVisitorFaker cv = new ClassVisitorFaker();
63
ClassReader cr = new ClassReader(getClassLoader().getResourceAsStream(clsRsName));
64
cr.accept(cv, 0);
65
List<AnnotationNode> annotationsList = cv.getVisibleAnnotations();
66data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (null != annotationsList)
{
67data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
for (Iterator<AnnotationNode> it = annotationsList.iterator(); it.hasNext();)
{
68
AnnotationNode annotation = it.next();
69
Type t = Type.getType(annotation.desc);
70data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (AnnotationFaker.class.getName().equals(t.getClassName()))
{
71
Class clazz = Class.forName(filenameToClassname(clsRsName));
72
ClassFaker faker = (ClassFaker) clazz.newInstance();
73
faker.hello();
74
}
75
}
76
}
77data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
78
}
79data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
80
}
81data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
82
private static void handleDirectory(final Set<String> result, final File file, final String path)
83data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
throws ZipException, IOException
{
84data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
for (final File child : file.listFiles())
{
85
final String newPath = path == null ? child.getName() : path + '/' + child.getName();
86data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (child.isDirectory())
{
87
handleDirectory(result, child, newPath);
88data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else if (child.getName().toLowerCase().endsWith(".jar"))
{
89
handleArchive(result, child);
90data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else
{
91
handleItem(result, newPath);
92
}
93
}
94
}
95data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
96data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
private static void handleItem(final Set<String> result, final String name)
{
97data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (name.endsWith(".class"))
{
98
result.add(name);
99
}
100
}
101data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
102data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
private static void handleArchive(final Set<String> result, final File file) throws ZipException, IOException
{
103
final ZipFile zip = new ZipFile(file);
104
final Enumeration<? extends ZipEntry> entries = zip.entries();
105data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
while (entries.hasMoreElements())
{
106
final ZipEntry entry = entries.nextElement();
107
final String name = entry.getName();
108
handleItem(result, name);
109
}
110
}
111data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
112data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
private static ClassLoader getClassLoader()
{
113
ClassLoader clsLoader = Thread.currentThread().getContextClassLoader();
114data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if (clsLoader == null)
{
115
clsLoader = ClassLoader.getSystemClassLoader();
116
}
117
return clsLoader;
118
}
119data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
120data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public static String filenameToClassname(final String filename)
{
121
return filename.substring(0, filename.lastIndexOf(".class")).replace('/', '.').replace('\\', '.');
122
}
123data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
124
}
125data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""