- 浏览: 84042 次
文章分类
- 全部博客 (136)
- 我的技术资料收集 (98)
- 具体技术 (1)
- 的技术资料收集 (4)
- All Articles (1)
- 机器学习 Machine Learning (1)
- 网络编程 (1)
- java (2)
- ava (1)
- 零散技术 (1)
- C# (3)
- 技术资料收集 (1)
- CQRS (1)
- 数据库技术(MS SQL) (1)
- .Net微观世界 (1)
- Oracle SQL学习之路 (1)
- C/C++ (1)
- JS/JQ (1)
- Js封装的插件/实例/方法 (2)
- 敏捷个人 (2)
- Javascript (1)
- 程序设计---设计模式 (1)
- Bug (1)
- 未知分类 (1)
- 程序设计 (1)
- Sharepoint (1)
- Computer Graphic (1)
- IT产品 (1)
- [06]JS/jQuery (1)
- [07]Web开发 (1)
- .NET Solution (1)
- Android (3)
- 机器学习 (1)
- 系统框架设计 (1)
- Others (1)
- 算法 (1)
- 基于Oracle Logminer数据同步 (1)
- 网页设计 (1)
- 原创翻译 (1)
- EXTJS (1)
- Jqgrid (1)
- 云计算 (1)
最新评论
一、什么是ASM
ASM是一个java字节码操纵框架,它能被用来动态生成类或者增强既有类的功能。ASM 可以直接产生二进制 class 文件,也可以在类被加载入 Java 虚拟机之前动态改变类行为。Java class 被存储在严格格式定义的 .class文件里,这些类文件拥有足够的元数据来解析类中的所有元素:类名称、方法、属性以及 Java 字节码(指令)。ASM从类文件中读入信息后,能够改变类行为,分析类信息,甚至能够根据用户要求生成新类。
使用ASM框架需要导入asm的jar包,下载链接:asm-3.2.jar。
二、如何使用ASM
ASM框架中的核心类有以下几个:
① ClassReader:该类用来解析编译过的class字节码文件。
② ClassWriter:该类用来重新构建编译后的类,比如说修改类名、属性以及方法,甚至可以生成新的类的字节码文件。
③ ClassAdapter:该类也实现了ClassVisitor接口,它将对它的方法调用委托给另一个ClassVisitor对象。
示例1.通过asm生成类的字节码
1 package com.asm3;
2
3 import java.io.File;
4 import java.io.FileNotFoundException;
5 import java.io.FileOutputStream;
6 import java.io.IOException;
7
8 import org.objectweb.asm.ClassWriter;
9 import org.objectweb.asm.Opcodes;
10
11 /**
12 * 通过asm生成类的字节码
13 * @author Administrator
14 *
15 */
16 public class GeneratorClass {
17
18 public static void main(String[] args) throws IOException {
19 //生成一个类只需要ClassWriter组件即可
20 ClassWriter cw = new ClassWriter(0);
21 //通过visit方法确定类的头部信息
22 cw.visit(Opcodes.V1_5, Opcodes.ACC_PUBLIC+Opcodes.ACC_ABSTRACT+Opcodes.ACC_INTERFACE,
23 "com/asm3/Comparable", null, "java/lang/Object", new String[]{"com/asm3/Mesurable"});
24 //定义类的属性
25 cw.visitField(Opcodes.ACC_PUBLIC+Opcodes.ACC_FINAL+Opcodes.ACC_STATIC,
26 "LESS", "I", null, new Integer(-1)).visitEnd();
27 cw.visitField(Opcodes.ACC_PUBLIC+Opcodes.ACC_FINAL+Opcodes.ACC_STATIC,
28 "EQUAL", "I", null, new Integer(0)).visitEnd();
29 cw.visitField(Opcodes.ACC_PUBLIC+Opcodes.ACC_FINAL+Opcodes.ACC_STATIC,
30 "GREATER", "I", null, new Integer(1)).visitEnd();
31 //定义类的方法
32 cw.visitMethod(Opcodes.ACC_PUBLIC+Opcodes.ACC_ABSTRACT, "compareTo",
33 "(Ljava/lang/Object;)I", null, null).visitEnd();
34 cw.visitEnd(); //使cw类已经完成
35 //将cw转换成字节数组写到文件里面去
36 byte[] data = cw.toByteArray();
37 File file = new File("D://Comparable.class");
38 FileOutputStream fout = new FileOutputStream(file);
39 fout.write(data);
40 fout.close();
41 }
42 }
生成一个类的字节码文件只需要用到ClassWriter类即可,生成Comparable.class后用javap指令对其进行反编译:javap -c Comparable.class >test.txt ,编译后的结果如下:
1 public interface com.asm3.Comparable extends com.asm3.Mesurable {
2 public static final int LESS;
3
4 public static final int EQUAL;
5
6 public static final int GREATER;
7
8 public abstract int compareTo(java.lang.Object);
9 }
注:一个编译后的java类不包含package和import段,因此在class文件中所有的类型都使用的是全路径。
示例2.修改类的字节码文件
C.java
1 package com.asm5;
2
3 public class C {
4 public void m() throws InterruptedException{
5 Thread.sleep(100);
6 }
7 }
将C.java类的内容改为如下:
1 package com.asm5;
2
3 public class C {
4 public static long timer;
5 public void m() throws InterruptedException{
6 timer -= System.currentTimeMillis();
7 Thread.sleep(100);
8 timer += System.currentTimeMillis();
9 }
10 }
为了弄清楚ASM是如何实现的,我们先编译这两个类,然后比对它们的TraceClassVisitor的输出,我们可以发现如下的不同(粗体表示)
GETSTATIC C.timer : J
INVOKESTATIC java/lang/System.currentTimilis()J
LSUB
PUTSTATIC C.timer : J
LDC 100
INVOKESTATIC java/lang/Thread.sleep(J)V
GETSTATIC C.timer : J
INVOKESTATIC java/lang/System.currentTimilis()J
LADD
PUTSTATIC C.timer : J
RETURN
MAXSTACK=4
MAXLOCALS=1
通过比对上面的指令,我们可以发现必须在m()方法的最前面增加四条指令,在RETURN指令前也增加四条指令,同时这四条必须位于xRETURN和ATHROW之前,因为这些指令都会结束方法的执行。
具体代码如下:
AddTimeClassAdapter.java
1 package com.asm5;
2
3 import org.objectweb.asm.ClassAdapter;
4 import org.objectweb.asm.ClassVisitor;
5 import org.objectweb.asm.FieldVisitor;
6 import org.objectweb.asm.MethodAdapter;
7 import org.objectweb.asm.MethodVisitor;
8 import org.objectweb.asm.Opcodes;
9
10 public class AddTimeClassAdapter extends ClassAdapter {
11 private String owner;
12 private boolean isInterface;
13 public AddTimeClassAdapter(ClassVisitor cv) {
14 super(cv);
15 }
16 @Override
17 public void visit(int version, int access, String name, String signature,
18 String superName, String[] interfaces) {
19 cv.visit(version, access, name, signature, superName, interfaces);
20 owner = name;
21 isInterface = (access & Opcodes.ACC_INTERFACE) != 0;
22 }
23 @Override
24 public MethodVisitor visitMethod(int access, String name, String desc,
25 String signature, String[] exceptions) {
26 MethodVisitor mv = cv.visitMethod(access, name, desc, signature, exceptions);
27 if(!name.equals("<init>") && !isInterface && mv!=null){
28 //为方法添加计时功能
29 mv = new AddTimeMethodAdapter(mv);
30 }
31 return mv;
32 }
33 @Override
34 public void visitEnd() {
35 //添加字段
36 if(!isInterface){
37 FieldVisitor fv = cv.visitField(Opcodes.ACC_PUBLIC+Opcodes.ACC_STATIC, "timer", "J", null, null);
38 if(fv!=null){
39 fv.visitEnd();
40 }
41 }
42 cv.visitEnd();
43 }
44
45 class AddTimeMethodAdapter extends MethodAdapter{
46 public AddTimeMethodAdapter(MethodVisitor mv) {
47 super(mv);
48 }
49 @Override
50 public void visitCode() {
51 mv.visitCode();
52 mv.visitFieldInsn(Opcodes.GETSTATIC, owner, "timer", "J");
53 mv.visitMethodInsn(Opcodes.INVOKESTATIC, "java/lang/System", "currentTimeMillis", "()J");
54 mv.visitInsn(Opcodes.LSUB);
55 mv.visitFieldInsn(Opcodes.PUTSTATIC, owner, "timer", "J");
56 }
57 @Override
58 public void visitInsn(int opcode) {
59 if((opcode>=Opcodes.IRETURN && opcode<=Opcodes.RETURN) || opcode==Opcodes.ATHROW){
60 mv.visitFieldInsn(Opcodes.GETSTATIC, owner, "timer", "J");
61 mv.visitMethodInsn(Opcodes.INVOKESTATIC, "java/lang/System", "currentTimeMillis", "()J");
62 mv.visitInsn(Opcodes.LADD);
63 mv.visitFieldInsn(Opcodes.PUTSTATIC, owner, "timer", "J");
64 }
65 mv.visitInsn(opcode);
66 }
67 @Override
68 public void visitMaxs(int maxStack, int maxLocal) {
69 mv.visitMaxs(maxStack+4, maxLocal);
70 }
71 }
72
73 }
Generator.java
1 package com.asm5;
2
3 import java.io.File;
4 import java.io.FileNotFoundException;
5 import java.io.FileOutputStream;
6 import java.io.IOException;
7
8 import org.objectweb.asm.ClassAdapter;
9 import org.objectweb.asm.ClassReader;
10 import org.objectweb.asm.ClassWriter;
11
12
13
14 public class Generator {
15
16 public static void main(String[] args){
17 try {
18 ClassReader cr = new ClassReader("com/asm5/C");
19 ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS);
20 ClassAdapter classAdapter = new AddTimeClassAdapter(cw);
21 //使给定的访问者访问Java类的ClassReader
22 cr.accept(classAdapter, ClassReader.SKIP_DEBUG);
23 byte[] data = cw.toByteArray();
24 File file = new File(System.getProperty("user.dir") + "\\WebRoot\\WEB-INF\\classes\\com\\asm5\\C.class");
25 FileOutputStream fout = new FileOutputStream(file);
26 fout.write(data);
27 fout.close();
28 System.out.println("success!");
29 } catch (FileNotFoundException e) {
30 e.printStackTrace();
31 } catch (IOException e) {
32 e.printStackTrace();
33 }
34 }
35
36 }
下面是一个测试类:
1 package com.asm5;
2
3 public class Test {
4 public static void main(String[] args) throws InterruptedException, NoSuchFieldException, SecurityException, IllegalArgumentException, IllegalAccessException {
5 C c = new C();
6 c.m();
7 Class cc = c.getClass();
8 System.out.println(cc.getField("timer").get(c));
9 }
10 }
输出结果为:100
未完待续.......
发表评论
-
C#WebBrowser控件使用教程与技巧收集--苏飞收集 - sufeinet
2013-06-28 12:07 1070原帖地址:http://www.cnblogs.com/suf ... -
我要喷一个自认为很垃圾的网站架构 - 老赵【苏州】
2013-06-28 12:01 1131原帖地址:http://www.cnblogs.com/lao ... -
[翻译] Oracle Database 12c 新特性Multitenant - Cheney Shue
2013-06-28 11:43 622原帖地址:http://www.cnblogs.com/ese ... -
memcahd 命令操作详解 - 阿正-WEB
2013-06-28 11:37 473原帖地址:http://www.cnblogs.com/azh ... -
面向过程的代码符合大众的思维方式吗? - 史蒂芬.王
2013-06-27 10:28 597原帖地址:http://www.cnblogs.com/ste ... -
面向过程的代码符合大众的思维方式吗? - 史蒂芬.王
2013-06-27 10:28 559原帖地址:http://www.cnblogs.com/ste ... -
RPG游戏之组队测试 - zthua
2013-06-27 10:22 560原帖地址:http://www.cnblogs.com/zth ... -
IT人们给个建议 - SOUTHER
2013-06-26 14:06 525原帖地址:http://www.cnblogs.com/sou ... -
Java向前引用容易出错的地方 - 银河使者
2013-06-26 14:00 495原帖地址:http://www.cnblogs.com/nok ... -
使用Func<T1, T2, TResult> 委托返回匿名对象 - 灰身
2013-06-26 13:54 800原帖地址:http://www.cnblo ... -
【web前端面试题整理03】来看一点CSS相关的吧 - 叶小钗
2013-06-25 10:45 786原帖地址:http://www.cnblogs.com/yex ... -
Windows 8 动手实验系列教程 实验6:设置和首选项 - zigzagPath
2013-06-25 10:27 622原帖地址:http://www.cnblogs.com/zig ... -
闲聊可穿戴设备 - shawn.xie
2013-06-25 10:21 567原帖地址:http://www.cnblo ... -
CentOS下Mysql安装教程 - 小学徒V
2013-06-23 15:24 610原帖地址:http://www.cnblogs.com/xia ... -
vmware安装ubuntu12.04嵌套安装xen server(实现嵌套虚拟化) - skyme
2013-06-23 15:18 838原帖地址:http://www.cnblogs.com/sky ... -
之前专门为IE6、7开发的网站如何迁移到IE10及可能遇到的问题和相应解决方案汇总 - 海之澜
2013-06-23 15:12 955原帖地址:http://www.cnblogs.com/wuz ... -
Android学习笔记--解析XML之SAX - 承香墨影
2013-06-23 15:01 411原帖地址:http://www.cnblo ... -
SQL Server 性能优化之——T-SQL TVF和标量函数
2013-06-19 09:32 674原帖地址:http://www.cnblogs.com/Boy ... -
Nginx学习笔记(二) Nginx--connection&request
2013-06-19 09:26 671原帖地址:http://www.cnblogs.com/cod ... -
从郭美美霸气侧漏看项目管理之项目经理防身术
2013-06-19 09:20 504原帖地址:http://www.cnblogs.com/had ...
相关推荐
提到的“mastering-java-bytecode.pdf”可能是一本关于Java字节码和ASM的教程,它可能涵盖了字节码的结构、ASM库的使用方法、实际案例分析等内容,是深入理解这两个主题的好资源。 7. **应用实例**: - Spring ...
ASM Java字节码操作框架PPT,结合已有AOP实现方法,对比所有对Java字节码操作方法做比较
在Java开发中,字节码框架ASM提供了一种强大的工具,允许程序员在运行时动态生成或修改类。ASM是一个底层的库,它直接操作Java字节码,这使得开发者能够在运行时改变类的行为或创建新的类。本文将深入探讨ASM框架的...
ASM 是一个 Java 字节码操控框架。它能被用来动态生成类或者增强既有类的功能。ASM 可以直接产生二进制 class 文件,也可以在类被加载入 Java 虚拟机之前动态改变类行为。Java class 被存储在严格格式定义的 .class ...
Java字节码框架ASM是一个强大的库,它允许程序员在运行时动态生成和修改Java类和接口的字节码。ASM提供了对JVM字节码的底层访问,这使得开发者能够实现诸如AOP(面向切面编程)或者元编程等高级功能。 首先,我们...
ASM是一个开源的Java字节码操控和分析框架,它能够用来动态生成类或者增强已有类的功能。ASM可以被用来创建Java代理、实现元编程、甚至深入到Java虚拟机(JVM)层面进行性能优化。在Java开发中,ASM库允许我们直接...
CGlib库利用ASM库(一个底层的Java字节码操作和分析框架)来生成和修改字节码。通过这种方式,开发者可以在程序运行时动态地创建新的类或修改现有类的行为,而无需重新编译源代码。 **动态代理** CGlib常用于实现...
ASM是一个低级别的Java字节码操控和分析框架,它可以直接生成和修改Java类的字节码。字节码是Java虚拟机(JVM)理解的机器语言,由.class文件中的二进制数据组成。ASM提供了一套API,使得开发者能够以抽象语法树...
以下是关于Java字节码的详细知识点: 1. **字节码的概念**:Java源代码在编译后生成的是类文件,每个类文件包含字节码,它是二进制形式的指令集,JVM能直接执行。这种设计允许Java程序在任何安装了JVM的平台上运行...
ObjectWeb ASM是一个轻量级且强大的Java字节码操控和分析框架,它允许程序员动态生成和修改类的字节码。本文将深入探讨如何使用ASM库来构建一个Method Monitor,以便监控和记录应用程序中方法的调用。 首先,了解...
**JAVA字节码操作库 BCEL** BCEL(Byte Code Engineering Library)是Java开发的一个重要工具,主要用于处理Java字节码。它为开发者提供了一种深入理解与操作Java类文件的底层机制,允许分析、创建、修改和优化字节...
- **版本信息**: ASM 4.0 是一个 Java 字节码操作和分析框架,由 Eric Bruneton 开发并维护,版权归属 Eric Bruneton(2007-2011)。 - **许可证**: 使用者可自由地重新发布和使用该文档的源码(LYX 格式)或编译后...
在Android应用开发中,无痕埋点是一种广泛用于数据分析的技术,它能够自动追踪用户在App中的行为,...通过学习和实践ASM插桩,开发者不仅可以提升应用的用户体验,还能更好地理解和掌握Java字节码及动态编程的精髓。
本文将深入探讨Java锁机制,并基于提供的"面向Java锁机制的字节码自动重构框架"来讨论其背后的原理和应用。 在Java中,锁主要分为内置锁(也称为监视器锁)和显式锁。内置锁是通过synchronized关键字实现的,它提供...
ASM是一个Java字节码操控和分析框架。它可以直接生成和修改Java类的字节码,从而实现诸如动态代理、代码混淆、性能监控等功能。在这个插件中,ASM被用来在运行时动态插入代码,记录并打印指定类或注解方法的执行时间...
通过学习如何使用ASM-Bytecode工具,开发者可以更好地理解Java字节码的结构,这对于理解JVM的工作原理、优化代码、或者实现类似AOP的动态功能都至关重要。ASM库提供了低级别的字节码操作能力,使得开发者可以直接与...
ASM 是一个 Java 字节码操控和分析框架,它可以直接生成和修改 Java 类和数组的字节码。ASM 提供了一种底层、高性能的方式来动态生成和解析 Java 类,广泛用于代理生成、AOP(面向切面编程)、性能监控以及字节码...
然而,这也意味着ASM的学习曲线相对较陡峭,需要一定的Java字节码基础。 在JavaAgent的上下文中,我们可以使用这两种工具在类加载到 JVM(Java 虚拟机)之前或之后对其进行增强。预加载增强(premain)常用于全局的...
ASM4是中国Java开发者常用的一款字节码操作框架ASM的第四个主要版本,它主要用于动态生成和分析Java字节码。ASM是一个低级别的库,可以直接操作和生成类的字节码,这在创建编译器、代码分析工具以及运行时代码修改等...