`
shihuan830619
  • 浏览: 584515 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Java字节码实现Aop(续二)

    博客分类:
  • asm
阅读更多
以下代码需要asm-5.0.2.jar和asm-commons-5.0.2.jar两个包。

package com.shihuan.field;

public class AccountField {

	String sql = "select * from tables";
	
	public void operation() {
        System.out.println("shihuan operation...");
    }
	
	int executeInternal() {
		
		System.out.println("shihuan executeInternal...");
		return 1;
	}
	
}


package com.shihuan.field;

public class TestAccountField {

	public static void main(String[] args) {
		AccountField t = new AccountField();
		t.operation();
		t.executeInternal();
	}

}


package com.shihuan.field;

public class SecurityCheckerField {
	
	public static void checkCode(AccountField afObj, java.util.Date startdate) {
		System.out.println("shihuan starttime --->> " + startdate);
		
		long endtime = System.currentTimeMillis();
		java.util.Date enddate = new java.util.Date(endtime);
		System.out.println("shihuan endtime --->> " + enddate);
		System.out.println(afObj.sql);
	}
	
}


package com.shihuan.field;

import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;

import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.ClassWriter;
import org.objectweb.asm.Label;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;

public class GeneratorAsmField {

	public static void main(String[] args) throws IOException {
		ClassReader cr = new ClassReader("com.shihuan.field.AccountField");
		ClassWriter cw = new ClassWriter(cr, ClassWriter.COMPUTE_MAXS);
		ClassVisitor cv = new ClassVisitor(Opcodes.ASM5, cw) {
			
			public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {
				MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions); // 先得到原始的方法
				MethodVisitor newMethod = mv;

				if (mv != null) {
					if ("executeInternal".equals(name)) { // 此处的executeInternal即为需要修改的方法,修改方法內容
						// 访问需要修改的方法
						newMethod = new MethodVisitor(Opcodes.ASM5, mv) {

							public void visitCode() {
								
								Label startlabel = new Label();
								visitLabel(startlabel);
								visitLineNumber(13, startlabel);
								
								
								visitMethodInsn(Opcodes.INVOKESTATIC, "java/lang/System", "currentTimeMillis", "()J", false);
								visitVarInsn(Opcodes.LSTORE, 2);
								Label ctmlabel = new Label();
								visitLabel(ctmlabel);
								visitLineNumber(14, ctmlabel);
								
								visitTypeInsn(Opcodes.NEW, "java/util/Date");
								visitInsn(Opcodes.DUP);
								visitVarInsn(Opcodes.LLOAD, 2);
								visitMethodInsn(Opcodes.INVOKESPECIAL, "java/util/Date", "<init>", "(J)V", false);
								visitVarInsn(Opcodes.ASTORE, 4);
								Label sdlabel = new Label();
								visitLabel(sdlabel);
								visitLineNumber(16, sdlabel);
								
								
								Label endlabel = new Label();
								visitLabel(endlabel);

visitLocalVariable("this", "Lcom/shihuan/field/AccountField;", null, startlabel, endlabel, 0);
								
								visitLocalVariable("starttime", "J", null, ctmlabel, endlabel, 2);
								visitLocalVariable("startdate", "Ljava/util/Date;", null, sdlabel, endlabel, 4);
								
								super.visitCode();
							}
							
							public void visitInsn(int opcode) {
								if ((opcode >= Opcodes.IRETURN && opcode <= Opcodes.RETURN) || opcode == Opcodes.ATHROW) {
									visitVarInsn(Opcodes.ALOAD, 0);
									visitVarInsn(Opcodes.ALOAD, 4);
									visitMethodInsn(
											Opcodes.INVOKESTATIC,
											"com/shihuan/field/SecurityCheckerField",
											"checkCode",
											"(Lcom/shihuan/field/AccountField;Ljava/util/Date;)V",
											false);
								}
								super.visitInsn(opcode);
							}
							
						};

					}
				}
				return newMethod;
			}

		};
		cr.accept(cv, ClassReader.SKIP_DEBUG);

		byte[] code = cw.toByteArray();
		OutputStream fos = new FileOutputStream("D:/myworkspace/JavaAsm/bin/com/shihuan/field/AccountField.class");
		fos.write(code);
		fos.close();
	}

}



【注】:本例中涉及到org.objectweb.asm.Label类和visitTypeInsn、visitInsn、visitVarInsn、visitLocalVariable等函数的参数意义问题,如果有想深入学习Asm5.0.2的朋友,请联系笔者,笔者学习Asm5.0.2的Java代码在shihuan83019@163.com的网盘的原创作品里。



----------------------------------------------------------------------------------

下面是笔者改Oracle的代码例子:
package oracle.jdbc.driver;

public class OracleMyEnd {

	public static void getMySqlInfo(OraclePreparedStatement opsObj, java.util.Date startdate){
		System.out.println(startdate);
		System.out.println("ShiHuan Is Inner --->> " + opsObj.sqlObject.originalSql);
	}
	
}


package oracle.jdbc.driver;

import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;

import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.ClassWriter;
import org.objectweb.asm.Label;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;

public class OracleGeneratorAsm {

	public static void main(String[] args) throws IOException {
		ClassReader cr = new ClassReader("oracle.jdbc.driver.OraclePreparedStatement");
		ClassWriter cw = new ClassWriter(cr, ClassWriter.COMPUTE_MAXS);
		ClassVisitor cv = new ClassVisitor(Opcodes.ASM5, cw) {

			
			public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {

				MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions); // 先得到原始的方法
				MethodVisitor newMethod = mv;

				if (mv != null) {
					if ("executeInternal".equals(name)) { // 此处的executeInternal即为需要修改的方法,修改方法內容
						// 访问需要修改的方法
						newMethod = new MethodVisitor(Opcodes.ASM5, mv) {

							public void visitCode() {
								
								Label startlabel = new Label();
								visitLabel(startlabel);
								visitLineNumber(3362, startlabel);
								
								
								visitMethodInsn(Opcodes.INVOKESTATIC, "java/lang/System", "currentTimeMillis", "()J", false);
								visitVarInsn(Opcodes.LSTORE, 5);
								Label ctmlabel = new Label();
								visitLabel(ctmlabel);
								visitLineNumber(3363, ctmlabel);
								
								visitTypeInsn(Opcodes.NEW, "java/util/Date");
								visitInsn(Opcodes.DUP);
								visitVarInsn(Opcodes.LLOAD, 5);
								visitMethodInsn(Opcodes.INVOKESPECIAL, "java/util/Date", "<init>", "(J)V", false);
								visitVarInsn(Opcodes.ASTORE, 7);
								Label sdlabel = new Label();
								visitLabel(sdlabel);
								visitLineNumber(3364, sdlabel);
								
								
								Label endlabel = new Label();
								visitLabel(endlabel);
								
								visitLocalVariable("this", "Loracle/jdbc/driver/OraclePreparedStatement;", null, startlabel, endlabel, 0);
								visitLocalVariable("starttime", "J", null, ctmlabel, endlabel, 5);
								visitLocalVariable("startdate", "Ljava/util/Date;", null, sdlabel, endlabel, 7);
								
								
								super.visitCode();
								
							}
							
							
							public void visitInsn(int opcode) {
								if ((opcode >= Opcodes.IRETURN && opcode <= Opcodes.RETURN) || opcode == Opcodes.ATHROW) {
									visitVarInsn(Opcodes.ALOAD, 0);
									visitVarInsn(Opcodes.ALOAD, 7);
									visitMethodInsn(
											Opcodes.INVOKESTATIC,
											"oracle/jdbc/driver/OracleMyEnd",
											"getMySqlInfo",
											"(Loracle/jdbc/driver/OraclePreparedStatement;Ljava/util/Date;)V",
											false);
								}
								super.visitInsn(opcode);
							}
							
						};

						
					}
				}
				return newMethod;
			}

		};
		cr.accept(cv, ClassReader.SKIP_DEBUG);

		byte[] code = cw.toByteArray();
		OutputStream fos = new FileOutputStream("D:/ojdbc6_b/oracle/jdbc/driver/OraclePreparedStatement.class");
		fos.write(code);
		fos.close();
	}

}
分享到:
评论

相关推荐

    Java字节码实现Aop

    Java字节码实现AOP(面向切面编程)是一种在程序运行时动态插入代码的技术,它使得我们可以在不修改原有代码的情况下,增加新的功能或监控已有功能。在Java中,AOP通常通过代理模式和字节码操作来实现,如Spring AOP...

    JavaEE CGLIB字节码增强方式实现AOP编程

    ASM是一个底层的Java字节码操控和分析框架,可以用来动态生成类或者增强已有类的功能。 **CGLIB的使用步骤**: 1. 引入依赖:在项目中引入CGLIB的依赖库,通常通过Maven或Gradle等构建工具添加。 2. 创建Enhancer...

    java字节码编辑器

    Java字节码编辑器是一种工具,它允许开发者直接编辑Java程序编译后的`.class`文件,而不是反编译后再重新编译。这种编辑器对于理解、调试和优化Java代码非常有用,尤其是对于那些无法访问源代码或者需要进行底层操作...

    JAVA字节码操作库 BCEL

    1. **字节码**:Java字节码是Java虚拟机(JVM)运行的二进制代码,它由javac编译器将源代码编译成。每个类都有一个`.class`文件,其中包含了该类的字节码。 2. **API组件**:BCEL包含一系列接口和类,如`ClassGen`,...

    cglib,字节码生成库是生成和转换Java字节码的高级API。它被aop、测试、数据访问框架用来生成动态代理对象和拦截字段访问。.zip

    **CGlib:高级Java字节码生成库** CGlib(Code Generation Library)是一个强大的高性能的代码生成库,它主要用于在运行期扩展Java类与实现Java接口。这个库最初由Evan Schooler创建,后来成为Apache软件基金会的一...

    从一个class文件深入理解Java字节码结构

    通过深入了解这些结构,开发者可以更好地理解Java字节码的工作原理,从而优化代码执行效率,调试程序,甚至实现字节码级别的动态修改和插桩技术。这对于性能调优、AOP(面向切面编程)以及理解JVM的工作方式至关重要...

    该项目主要是JAVA字节码的学习项目.zip

    6. **字节码操纵**:Java字节码操纵库,如ASM、BCEL、CGLIB等,允许程序员在运行时动态修改类的字节码,实现AOP(面向切面编程)和代码注入等功能。 7. **安全性**:Java的字节码验证机制确保了字节码的安全性,...

    基于Java字节码增强技术实现的协程Demo.zip

    Java字节码增强技术是Java开发中的一种高级技巧,它允许开发者在程序运行时动态修改类的字节码,从而实现一些特殊的功能,比如AOP(面向切面编程)、性能监控、事务管理等。协程是一种轻量级的线程,它在执行过程中...

    深入字节码 -- 使用 ASM 实现 AOP1

    Java 字节码理解为 JVM 能够识别的机器语言,它是 Java 源代码经过编译后的产物。每个字节码指令对应一个特定的操作,比如 `getstatic` 用于获取静态字段,`ldc` 加载常量,`invokevirtual` 调用虚方法等。JVM 会...

    Java字节码和asm入门资料

    Java字节码是Java虚拟机...总的来说,Java字节码和ASM是Java开发中的高级主题,它们对于理解Java运行机制、实现高性能的代码以及进行动态编程至关重要。通过学习和实践,开发者可以掌握这些技术,提升自己的编程能力。

    ASM Java字节码操作框架

    ASM Java字节码操作框架PPT,结合已有AOP实现方法,对比所有对Java字节码操作方法做比较

    java字节码例子 可以动态修改类 bcel

    Java字节码是Java平台的...通过学习这些内容,开发者可以掌握Java字节码的基本操作,并利用BCEL在实际项目中实现高级功能。同时,这也是一种深入理解JVM工作原理的方式,有助于提升Java编程的技能和对底层机制的理解。

    javassist,Java字节码工程工具包.zip

    Java字节码工程工具包,通常被称为Javassist,是一个开源项目,专为Java开发者提供了一种方便的方式来处理和操作字节码。Javassist在Java应用开发中扮演着重要的角色,尤其是在动态代理、AOP(面向切面编程)以及...

    ASM操作字节码,动态生成Java类class文件

    通过这样的方式,ASM可以帮助我们深入理解Java字节码,同时也为我们提供了强大的工具来实现动态代码生成,这对于实现诸如AOP这样的高级编程模式至关重要。在实际项目中,ASM通常与其他工具结合使用,如CGLIB或Byte...

    java分页 动态代理 aop实现

    2. CGLIB库:当目标类没有实现接口时,CGLIB库可以通过字节码操作生成子类实现动态代理。 现在,我们转向"AOP(面向切面编程)"。AOP是Spring框架的核心特性之一,它允许我们将关注点(如日志、事务管理、安全检查...

    java字节码java字节码

    ### Java字节码详解 #### 一、Java字节码简介 Java字节码是一种中间语言,由Java编译器生成,在Java虚拟机(JVM)上运行。它是一种平台无关的二进制代码格式,使得Java程序可以在任何支持JVM的平台上运行而无需重新...

    JAVA字节码操作库 BCEL.7z

    **JAVA字节码操作库 BCEL** BCEL(Byte Code Engineering Library)是Java平台上的一款开源库,主要用于处理Java字节码。它提供了对字节码的解析、创建、修改和分析的能力,使得开发者能够深入理解并操作Java类文件...

    java-用Java动态代理实现AOP.pdf

    对于有具体实现类的情况,可能需要使用其他AOP框架,比如Spring AOP,它基于字节码操作,能处理接口和类,功能更加强大和灵活。 总之,Java动态代理提供了一种在运行时创建代理类并实现AOP的方法,它可以帮助我们...

    JAVA字节码编辑工具.zip

    JAVA字节码编辑工具,正如其标题所示,是一种专门用于编辑Java字节码的软件。在Java编程语言中,源代码被编译成字节码(.class文件),这是一种平台无关的中间表示形式,可以在Java虚拟机(JVM)上运行。这个工具...

    java源码:JAVA字节码操作库 BCEL.zip

    Java字节码是Java虚拟机(JVM)执行的二进制代码,它是由Java源代码编译而成。每条字节码指令都是一个单字节的操作码,后面可能跟随零个或多个操作数。字节码是平台无关的,使得Java应用程序可以在任何支持JVM的设备...

Global site tag (gtag.js) - Google Analytics