【Javassist】快速入门系列05 当有指定方法调用时替换方法调用的内容

系列文章目录

01 在方法体的开头或结尾插入代码
02 使用Javassist实现方法执行时间统计
03 使用Javassist实现方法异常处理
04 使用Javassist更改整个方法体
05 当有指定方法调用时替换方法调用的内容



前言

上一章我们介绍了使用Javassist更改整个方法体,学会了Javassist的setBody()方法使用。本章主要介绍当检测到指定方法调用时替换方法调用的内容,method.instrument方法的使用。


引入Javassist jar包

在上几篇文章已经引入了javassist的jar包,如果你是第一次观看本系列文章,也可以复制以下maven依赖将jar包导入工程。

        <dependency>
            <groupId>org.javassist</groupId>
            <artifactId>javassist</artifactId>
            <version>3.28.0-GA</version>
        </dependency>

当有指定方法调用时替换方法调用的内容

/**
 * 【Javassist】快速入门系列05 当有指定方法调用时替换方法调用的内容
 * 公众号&B站:精致的王同学
 * @author 精致的王同学
 * @date 2022/12/21 16:25
 */
public class Basic05MethodCall {
    public static void main(String[] args) throws Exception{
        // 获取javassist默认类池
        ClassPool pool = ClassPool.getDefault();
        // 获取basic.Basic05Test的CtClass对象
        CtClass ctClass = pool.get("basic.Basic05Test");
        // 获取Basic05Test类的main方法
        CtMethod method = ctClass.getDeclaredMethod("main");
        // 更改main方法的方法体
        method.instrument(new ExprEditor(){
            @Override
            public void edit(MethodCall m) throws CannotCompileException {
                if (m.getClassName().equals("basic.StockService") && m.getMethodName().equals("queryStockNum")) {
                    m.replace("{System.out.println(\"MethodCall\");$_ = $proceed($$);}");
                }
            }
        });
        // 将类写成文件
        ctClass.writeFile();
        // 获取basic.Basic05Test的clazz对象
        Class<?> clazz = ctClass.toClass();
        // 获取修改后的basic.Basic05Test的实例
        Object obj = clazz.newInstance();
        // 获取修改后的main方法
        Method main = clazz.getDeclaredMethod("main",String[].class);
        // 模拟调用修改后的main方法
        main.invoke(obj,(Object) new String[0]);
    }
}

以上Basic05MethodCall类创建了一个main方法,该方法中首先获取javassist的类池pool,然后调用pool.get(“basic.Basic05Test”)方法获取到basic包下的Basic05Test类。Basic05Test类源码如下:

/**
 * 第5节测试类
 * 公众号&B站:精致的王同学
 * @author 精致的王同学
 * @date 2022/12/21 17:24
 */
public class Basic05Test {
    public static void main(String[] args) throws Exception{
        StockService stockService = new StockService();
        stockService.queryStockNum(1l);
    }
}

该类中调用了StockService 的stockService方法。StockService 源码如下:

/**
 * 库存业务类
 * 公众号&B站:精致的王同学
 * @author 精致的王同学
 * @date 2022/12/19 17:25
 */
public class StockService {
    public Integer queryStockNum(Long skuId) throws Exception {
        // 调用库存数量接口
        Random random = new Random();
        Long mills = Long.valueOf(random.nextInt(5)*1000);
        // 模拟接口调用耗时
        Thread.sleep(mills);
        return 0;
    }
}

StockService 的queryStockNum方法用于模拟调用获取sku库存数量接口的调用。

回到Basic05MethodCall 的main方法,在获取到Basic05Test类的ctClass的对象之后,获取其main方法的方法对象。然后调用method.instrument(ExprEditor editor)方法搜索类型为方法调用的语句。

instrument方法接收一个ExprEditor 类型的对象,该类有很多重载的edit方法,其中参数为MethodCall 的重载方法代表搜索method方法内的方法调用。然后判断如果是basic.StockService类的queryStockNum方法调用,则将此方法调用替换为指定代码块的内容。

ExprEditor 类重载的edit方法如下:
ExprEditor 类重载的edit方法

如果指定方法的调用有返回值的话,必须指定其内容。$_ 代表方法调用的返回值。

$proceed 代表调用指定的方法,本例中为queryStockNum方法。

$$ 代表方法的全部参数。

new ExprEditor() 代码块中的内容所做的事情为搜索main方法中所有的方法调用,当方法调用是basic.StockService类的queryStockNum方法时在屏幕上打印MethodCall。

最后模拟调用修改后的main方法结果如下:
main方法结果

总结

本篇文章介绍了使用Javassist当有指定方法调用时替换方法调用的内容,学习了 method.instrument的用法。以及参数为MethodCall 的重载方法的含义。

说明

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Javassist中,ClassPool的insertClassPath方法用于向类路径中添加新的路径。通过调用insertClassPath方法,你可以告诉Javassist在查找类文件搜索的路径范围。 具体来说,insertClassPath方法的作用是将指定的路径添加到ClassPool的搜索路径列表中。当Javassist需要加载类文件,它会按照添加路径的顺序去搜索类文件。如果类文件在指定的路径中找到,Javassist就可以成功加载该类。 insertClassPath方法支持不同类型的路径,包括本地文件系统路径、JAR文件路径、目录路径等。你可以根据需要选择合适的路径类型。 下面是使用insertClassPath方法的示例: ```java import javassist.ClassPool; public class Example { public static void main(String[] args) { ClassPool pool = ClassPool.getDefault(); // 添加本地文件系统路径 pool.insertClassPath("/path/to/classes"); // 添加JAR文件路径 pool.insertClassPath("/path/to/library.jar"); // 添加目录路径 pool.insertClassPath("/path/to/directory"); // 加载并操作类 // ... } } ``` 在上述示例中,我们使用insertClassPath方法分别添加了本地文件系统路径、JAR文件路径和目录路径。当需要加载类Javassist会按照添加路径的顺序进行搜索。你可以根据需要多次调用insertClassPath方法来添加多个路径。 通过使用insertClassPath方法,你可以确保Javassist能够正确地找到需要操作的类,从而进行类的动态修改和增强等操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值