系列文章目录
01 在方法体的开头或结尾插入代码
 02 使用Javassist实现方法执行时间统计
 03 使用Javassist实现方法异常处理
 04 使用Javassist更改整个方法体
 05 当有指定方法调用时替换方法调用的内容
 06 当有构造方法调用时替换方法调用的内容
 07 当检测到字段被访问时使用语句块替换访问
 08 当检测到对象(不包括数组)创建时用代码块替换
 09 当检测到对象(不包括数组)创建时用代码块替换
 10 当检测到instanceof表达式时用代码块替换
 11 当检测到显示类型转换时用代码块替换
 12 当检测到catch语句时在catch前插入代码
文章目录
- 系列文章目录
 - 前言
 - 引入Javassist jar包
 - 当检测到catch语句时在catch前插入代码
 
- 总结
 - 说明
 
前言
上一章我们介绍了当检测到显示类型转换时用代码块替换,学习了 method.instrument的用法。以及参数为Cast的重载方法edit的含义。本章主要介绍当检测到catch语句时在catch前插入代码。
引入Javassist jar包
在上几篇文章已经引入了javassist的jar包,如果你是第一次观看本系列文章,也可以复制以下maven依赖将jar包导入工程。
        <dependency>
            <groupId>org.javassist</groupId>
            <artifactId>javassist</artifactId>
            <version>3.28.0-GA</version>
        </dependency>
 
当检测到catch语句时在catch前插入代码
/**
 * 【Javassist】快速入门系列11 当检测到显示类型转换时用代码块替换
 * 公众号&B站:精致的王同学
 * @author 精致的王同学
 * @date 2022/12/28 18:18
 */
public class Basic12Handler {
    public static void main(String[] args) throws Exception{
        // 获取javassist默认类池
        ClassPool pool = ClassPool.getDefault();
        // 获取basic.Basic12Test的ctclass类
        CtClass ctClass = pool.get("basic.Basic12Test");
        // 获取basic.Basic12Test的main方法
        CtMethod method = ctClass.getDeclaredMethod("main");
        // 当检测到catch语句时在catch前插入代码
        method.instrument(new ExprEditor() {
            @Override
            public void edit(Handler h) throws CannotCompileException {
                try {
                    if (h.getType().equals(pool.get("java.lang.Exception"))) {
                        h.insertBefore("System.out.println(\"Handler\");");
                    }
                } catch (NotFoundException e) {
                    e.printStackTrace();
                }
            }
        });
        // 将类写成文件
        ctClass.writeFile();
        // 获取修改后的class文件
        Class<?> clazz = ctClass.toClass();
        // 获取修改后的类实例
        Object obj = clazz.newInstance();
        // 获取修改后的main方法
        Method main = clazz.getDeclaredMethod("main", String[].class);
        // 调用修改后的main方法
        main.invoke(obj,(Object) new String[0]);
    }
}
 
以上Basic12Handler 类创建了一个main方法,该方法中首先获取javassist的类池pool,然后调用pool.get(“basic.Basic12Test”)方法获取到basic包下的Basic12Test类。Basic12Test类源码如下:
/**
 * 第12节测试类
 * 公众号&B站:精致的王同学
 * @author 精致的王同学
 * @date 2022/12/28 17:54
 */
public class Basic12Test {
    public static void main(String[] args) {
        SkuService skuService = new SkuService();
        try {
            skuService.getSkuProfitRate(1L);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
 
该类中有一个main方法,创建了一个skuService 对象。然后将调用其getSkuProfitRate方法。如果有异常则打印异常,SkuService 类源码如下:
/**
 * 商品业务类
 *
 * @author 精致的王同学
 * @date 2022/12/19 23:38
 */
public class SkuService {
    public BigDecimal getSkuProfitRate(Long skuId){
        // 模拟调用查价格接口获取商品原价和商品销售价
        BigDecimal originalPrice = null;
        originalPrice = getOriginalPrice(skuId);
        BigDecimal salePrice = getSalePrice(skuId);
        BigDecimal profit = salePrice.subtract(originalPrice);
        return profit.divide(originalPrice);
    }
    private BigDecimal getSalePrice(Long skuId) {
        return new BigDecimal(130.00);
    }
    private BigDecimal getOriginalPrice(Long skuId) {
        // 模拟sku原始价格为0,价格未初始化的情况
        BigDecimal originalPrice = new BigDecimal(0.00);
        if (BigDecimal.ZERO.compareTo(originalPrice) == 0) {
            throw new ErrorCodeException(10001,"sku原始价格不能等于0!");
        }
        return originalPrice;
    }
}
 
SkuService类的getSkuProfitRate方法根据skuId获取其sku利润率。其中调用getOriginalPrice方法抛出一个异常。
回到Basic12Test的main方法,在获取到Basic12Test类的ctClass的对象之后,获取其main方法的方法对象。
接着调用method.instrument(ExprEditor editor)方法搜索method内的catch语句。调用h.getType()方法获取异常类型的ctclas文件判断是否是java.lang.Exception类型的,如果是则调用h.insertBefore(“System.out.println(“Handler”);”);在catch前插入代码。
instrument方法接收一个ExprEditor 类型的对象,该类有很多重载的edit方法,其中参数为Handler的重载方法代表搜索方法内的catch语句。
在replace的代码块中,以下符号有特殊含义:
$1  catch子句捕获的异常对象。
$r  catch子句捕获的异常的类型。它用于强制转换表达式中。
$w  包装器类型。它用于强制转换表达式中。
$type  java.lang.Class对象表示catch子句捕获的异常的类型。
 
最后模拟调用修改后的main方法结果如下:
 
总结
本篇文章介绍了使用Javassist当检测到catch语句时在catch前插入代码,学习了 method.instrument的用法。以及参数为Handler的重载方法edit的含义。



















