JAVA_javassist学习

本文深入探讨了Java字节码操作的概念,重点介绍了Javassist库的使用,包括动态生成类、方法、属性,以及修改现有类和方法的能力。通过实例展示了如何利用Javassist进行代码增强和重构。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

javassist
用于对java字节码文件进行操作,可以动态生成新的类、方法、添加属性等,亦可对已有类、方法、属性进行修改

需要导入的jar
javassist.jar

1.使用javassist生成一个新的类
package com.bjsxt.test;

import javassist.CannotCompileException;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtConstructor;
import javassist.CtField;
import javassist.CtMember;
import javassist.CtMethod;

/**
 * 测试使用javassist生成一个新的类
 * @author WL20180732
 *
 */
public class Demo01 {
	public static void main(String[] args) throws Exception {
		ClassPool pool = ClassPool.getDefault();
		CtClass cc = pool.makeClass("com.bjsxt.bean.Emp");
		
		// 创建属性
		CtField f1 = CtField.make("private int empno;", cc);
		CtField f2 = CtField.make("private String ename;", cc);
		cc.addField(f1);
		cc.addField(f2);
		
		// 创建方法
		CtMethod m1 = CtMethod.make("public int getEmpno() {return this.empno;}", cc);
		CtMethod m2 = CtMethod.make("public void setEmpno(int empno) {this.empno = empno;}", cc);
		cc.addMethod(m1);
		cc.addMethod(m2);
		
		// 添加构造器
		CtConstructor constructor = new CtConstructor(
				new CtClass[]{CtClass.intType, pool.get("java.lang.String")}, cc);
		
		constructor.setBody("{this.empno = empno; this.ename = ename;}");
		cc.addConstructor(constructor);
		cc.writeFile("E:/mycode"); // 将上面构造好的类写入指定目录下
		System.out.println("类生成成功");
	}
}


在这里插入图片描述

2.测试javassist的API
package com.bjsxt.test;

import java.io.IOException;
import java.lang.reflect.Method;
import java.util.Arrays;

import javax.security.sasl.AuthorizeCallback;

import javassist.CannotCompileException;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtConstructor;
import javassist.CtField;
import javassist.CtMethod;
import javassist.CtNewMethod;
import javassist.Modifier;
import javassist.NotFoundException;

/**
 * 测试javassist的API
 * @author WL20180732
 *
 */
public class Demo02 {
	
	/**
	 * 处理类的基本用法
	 * @throws NotFoundException 
	 * @throws CannotCompileException 
	 * @throws IOException 
	 */
	public static void test01() throws Exception {
		ClassPool classPool = ClassPool.getDefault();
		CtClass cc = classPool.get("com.bjsxt.test.Emp");  // 获得已有类
		
		byte[] bytes = cc.toBytecode();
		System.out.println(Arrays.toString(bytes));
		
		System.out.println(cc.getName()); // 获取全类名
		System.out.println(cc.getSimpleName()); // 获取类名
		System.out.println(cc.getSuperclass()); // 获得父类
		System.out.println(cc.getInterfaces()); // 获取接口
	}
	
	/**
	 * 测试产生新的方法
	 * @throws Exception 
	 */
	public static void test02() throws Exception {
		ClassPool classPool = ClassPool.getDefault();
		CtClass cc = classPool.get("com.bjsxt.test.Emp");  // 获得已有类
		
//		CtMethod m = CtNewMethod.make("public int add(int a, int b) {return a+b;}", cc);
		CtMethod m = new CtMethod(
				CtClass.intType,   // 返回值类型
				"add", 
				new CtClass[]{CtClass.intType, CtClass.intType}, cc);
		
		m.setModifiers(Modifier.PUBLIC);  //设置方法修饰符
		m.setBody("{System.out.println($1 + $2); return $1 + $2;}");  // $0:this关键字,$$: 方法的所有参数,$1:方法的第一个参数,$2:方法的第二个参数...
		
		cc.addMethod(m);
		
		//通过反射调用生成的方法
		Class clazz = cc.toClass();
		Object obj = clazz.newInstance();  // 通过调用emp的无参构造,创建新的Emp对象
		Method method = clazz.getDeclaredMethod("add", int.class, int.class);
		Object result = method.invoke(obj, 200, 300);
		System.out.println(result);
	}
	
	/**
	 * 修改已有的方法信息
	 * @throws Exception
	 */
	public static void test03() throws Exception{
		ClassPool classPool = ClassPool.getDefault();
		CtClass cc = classPool.get("com.bjsxt.test.Emp");  // 获得已有类
		
		CtMethod cm = cc.getDeclaredMethod("sayHello", new CtClass[]{CtClass.intType});
		cm.insertBefore("System.out.println($1);System.out.println(\"start!!\");");
		cm.insertAfter("System.out.println(\"end!!\");");
		cm.insertAt(27, "System.out.println(\"lineNum=\" + 27);"); // 在某一行处加代码
		//通过反射调用生成的方法
		Class clazz = cc.toClass();
		Object obj = clazz.newInstance();  // 通过调用emp的无参构造,创建新的Emp对象
		Method method = clazz.getDeclaredMethod("sayHello", int.class);
		Object result = method.invoke(obj, 200);
		System.out.println(result);
	}
	
	/**
	 * 属性的操作
	 * @throws Exception
	 */
	public static void test04() throws Exception{
		ClassPool classPool = ClassPool.getDefault();
		CtClass cc = classPool.get("com.bjsxt.test.Emp");  // 获得已有类
		
//		CtField f1 = CtField.make("private int empno;", cc);
		CtField f1 = new CtField(CtClass.intType, "salary", cc);
		f1.setModifiers(Modifier.PRIVATE);
		cc.addField(f1);
		cc.addMethod(CtNewMethod.getter("getSalary", f1)); // 给f1属性添加getter方法
		cc.addMethod(CtNewMethod.getter("setSalary", f1)); // 给f1属性添加setter方法
//		cc.getDeclaredField("ename");  // 获取指定属性
		
		
	}
	
	/**
	 * 构造方法的操作
	 * @throws Exception
	 */
	public static void test05() throws Exception{
		ClassPool classPool = ClassPool.getDefault();
		CtClass cc = classPool.get("com.bjsxt.test.Emp");  // 获得已有类
		
		CtConstructor[] constructors = cc.getConstructors();
		for (CtConstructor ctConstructor : constructors) {
			System.out.println(ctConstructor.getLongName());
//			ctConstructor.insertBefore(src);
		}
	}
	
	public static void test06() throws Exception{
		ClassPool classPool = ClassPool.getDefault();
		CtClass cc = classPool.get("com.bjsxt.test.Emp");  // 获得已有类
		Object[] all = cc.getAnnotations();
		Author a = (Author) all[0];
		String name = a.name();
		int year = a.year();
		System.out.println("name="+name + ",year=" + year);
	}
	
	public static void main(String[] args) throws Exception {
//		test01();
//		test02();
//		test03();
//		test04();
//		test05();
		test06();
	}
}

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(value={ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface Author {
	String name();
	int year();
}

package com.bjsxt.test;

@Author(name="azor", year=2020)
public class Emp {
	
	private int empno;
	private String ename;
	public int getEmpno() {
		return empno;
	}
	public void setEmpno(int empno) {
		this.empno = empno;
	}
	public String getEname() {
		return ename;
	}
	public void setEname(String ename) {
		this.ename = ename;
	}
	public Emp() {
	}
	public Emp(int empno, String ename) {
		this.empno = empno;
		this.ename = ename;
	}
	
	public void sayHello(int a) {
		System.out.println("say Hello," + a);
	}
	
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值