第三篇:Spring之AOP前奏

一 提出需求

在项目中经常需要输出日志以定位问题,例如,在方法执行前执行后输出日志,如果不使用Spring的AOP功能该怎样实现呢?
情景:数学计算器
要求
①执行加减乘除运算
②日志:在程序执行期间追踪正在发生的活动
③验证:希望计算器只能处理正数的运算

1、常规操作
在代码中实现日志代码:
在这里插入图片描述
缺点:代码混乱,许多非业务需求代码加入。如果日志需求有变化则需要修改所有模块
2、使用动态代理
代理设计模式的原理:使用一个代理将对象包装起来,然后用该代理对象取代原始对象。任何对原始对象的调用都要通过代理。代理对象决定是否以及何时将方法调用转到原始对象上。
在这里插入图片描述
重新优化代码:
计算器接口为:

public interface ArithmeticCalculator {

	int add(int i, int j);
	int sub(int i, int j);
	int mul(int i, int j);
	int div(int i, int j);
	
}

实现计算器接口:

public class ArithmeticCalculatorImpl implements ArithmeticCalculator{
	@Override
	public int add(int i, int j) {
		int result = i + j;
		return result;
	}
	@Override
	public int sub(int i, int j) {
		int result = i - j;
		return result;
	}
	@Override
	public int mul(int i, int j) {
		int result = i * j;
		return result;
	}
	@Override
	public int div(int i, int j) {
		int result = i / j;
		return result;
	}

}

重点来了,书写动态代理类,实现动态代理接口InvocationHandler
※ java动态代理机制中有两个重要的类和接口InvocationHandler(接口)和Proxy(类),这一个类Proxy和接口InvocationHandler是我们实现动态代理的核心;

public class CalculatorLogingHandler implements InvocationHandler {

    /**要代理的对象*/
    private  Object target;

    public CalculatorLogingHandler(Object target) {
        this.target = target;
    }

    /**返回代理对象*/
    public  Object getLoggingProxy(){
        //Proxy动态代理核心类
        return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),new CalculatorLogingHandler(target));
    }

    /**
     * 实现InvocationHandler的invoke方法
     * @param proxy:代理对象。 一般不使用该对象,在方法内会造成StackOverflowError
     * @param method:正在被调用的方法
     * @param args:调用方法传入的参数
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //调用proxy任何方法都会造成StackOverflowError
       // proxy.toString();
        String methodName = method.getName();
        //在方法调用前调用日志
        System.out.println("[before] The method " + methodName + " begins with " + Arrays.asList(args));
        Object result = null;
        try {
            result = method.invoke(target, args);
        } catch (NullPointerException e) {
            e.printStackTrace();
            //异常通知, 可以访问到方法出现的异常
        }
        //方法结束后打印日志
        System.out.println("[after] The method ends with " + result);
        return result;
    }

}

验证代码:

public class Main {
    public static void main(String[] args) {
        ArithmeticCalculator arithmeticCalculator = new ArithmeticCalculatorImpl();
        CalculatorLogingHandler calculatorLogingHandler = new CalculatorLogingHandler(arithmeticCalculator);
        arithmeticCalculator =(ArithmeticCalculator) calculatorLogingHandler.getLoggingProxy();
        arithmeticCalculator.add(1,2);

    }
}

输出结果:
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值