系列文章目录
01 在方法体的开头或结尾插入代码
 02 使用Javassist实现方法执行时间统计
 03 使用Javassist实现方法异常处理
 04 使用Javassist更改整个方法体
 05 当有指定方法调用时替换方法调用的内容
 06 当有构造方法调用时替换方法调用的内容
 07 当检测到字段被访问时使用语句块替换访问
文章目录
- 系列文章目录
- 前言
- 引入Javassist jar包
- 当检测到字段被访问时使用语句块替换访问
 
- 总结
- 说明
前言
上一章我们介绍了当构造方法调用时替换方法调用的内容,学习了 method.instrument的用法。以及参数为ConstructorCall的重载方法的含义。本章主要介绍当检测到字段被访问时使用语句块替换访问。
引入Javassist jar包
在上几篇文章已经引入了javassist的jar包,如果你是第一次观看本系列文章,也可以复制以下maven依赖将jar包导入工程。
        <dependency>
            <groupId>org.javassist</groupId>
            <artifactId>javassist</artifactId>
            <version>3.28.0-GA</version>
        </dependency>
当检测到字段被访问时使用语句块替换访问
/**
 * 【Javassist】快速入门系列07 当检测到字段被访问时使用语句块替换访问
 * 公众号&B站:精致的王同学
 * @author wangfengxi1
 * @date 2022/12/25 18:02
 */
public class Basic07FieldAccess {
    public static void main(String[] args) throws Exception{
        // 获取javassist默认类池
        ClassPool pool = ClassPool.getDefault();
        // 获取basic.Basic07Test的ctClass对象
        CtClass ctClass = pool.get("basic.Basic07Test");
        // 获取basic.Basic07Test的cmain方法
        CtMethod method = ctClass.getDeclaredMethod("main");
        // 当检测到字段被访问时使用语句块替换访问
        method.instrument(new ExprEditor() {
            @Override
            public void edit(FieldAccess f) throws CannotCompileException {
                try {
                    if (f.getClassName().equals("basic.Basic07Test") && f.getFieldName().equals("name")) {
                        f.replace("System.out.println(\"FieldAccess\");$_=$proceed($$);");
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        });
        // 将类写成文件
        ctClass.writeFile();
        //获取basic.Basic07Test类修改后的class对象
        Class<?> clazz = ctClass.toClass();
        // 获取basic.Basic07Test类修改后的实例
        Object obj = clazz.newInstance();
        // 获取修改后的main方法
        Method main = clazz.getDeclaredMethod("main", String[].class);
        // 调用修改后的main方法
        main.invoke(obj,(Object) new String[0]);
    }
}
以上Basic07FieldAccess类创建了一个main方法,该方法中首先获取javassist的类池pool,然后调用pool.get(“basic.Basic07Test”)方法获取到basic包下的Basic07Test类。Basic07Test类源码如下:
/**
 * 第7节测试类
 *
 * @author wangfengxi1
 * @date 2022/12/25 20:16
 */
public class Basic07Test {
    private String name;
    public Basic07Test() {
    }
    public Basic07Test(String name) {
        this();
        this.name = name;
    }
    public static void main(String[] args) {
        Basic07Test basic07Test = new Basic07Test("小明");
        System.out.println(basic07Test.name);
    }
}
该类中有一个main方法创建了一个Basic07Test对象,然后调用System.out.println(basic07Test.name);输出姓名。这个语句访问了basic07Test.name
回到Basic07FieldAccess 的main方法,在获取到basic07Test类的ctClass的对象之后,获取其main方法的方法对象。
接着调用method.instrument(ExprEditor editor)方法搜索method内的字段访问。判断如果其类是basic.Basic07Test且被访问的字段是name则调用f.replace(“System.out.println(“FieldAccess”); = _= =proceed($$);”);方法替换字段的访问。
instrument方法接收一个ExprEditor 类型的对象,该类有很多重载的edit方法,其中参数为FieldAccess 的重载方法代表搜索方法内的字段访问。
如果表达式是读访问,则必须在源文本中为$_赋值。$_的类型是字段的类型。
在replace的代码块中,以下符号有特殊含义:
$0  包含表达式访问的字段的对象。这并不等同于this。this表示调用包含表达式的方法的对象。如果字段是静态的,则$0为空。
$1  如果表达式是写访问,则代表存储在字段中的值。否则,$1不可用。
$_  如果表达式是读取访问,则代表字段访问的结果值。否则,$_中存储的值将被丢弃。
$r  如果表达式是读取访问,则代表字段的类型。否则,$r无效。
$class  表示声明字段的类的java.lang.class对象。
$type  表示字段类型的java.lang.Class对象。
$proceed  执行原始字段访问的虚拟方法的名称。
最后模拟调用修改后的main方法结果如下:
 
总结
本篇文章介绍了使用Javassist当检测到字段被访问时使用语句块替换访问,学习了 method.instrument的用法。以及参数为FieldAccess 的重载方法edit的含义。



















