AOP-03

7.AOP-切入表达式

7.1切入表达式的具体使用

1.切入表达式的作用:

通过表达式的方式定义一个或多个具体的连接点。

2.语法细节:

(1)切入表达式的语法格式:

execution([权限修饰符] [返回值类型] [简单类名/全类名] [方法名]([参数列表])

若目标类、接口与该切面类在用同一个包中,可以省略包名,只写简单类名

(2)举例说明:

例子1:

表达式:execution(* com.sina.spring.ArithmeticCalculator.*(..))

含义:ArithmeticCalculator.* :接口中声明的所有方法。第一个 * 代表任意修饰符和任意返回值。 第二个 * 代表任意方法。.. 代表匹配任意数量和任意类型的参数,若目标类、接口与该切面类在用同一个包中,可以省略包名。

例子2:

表达式:execution(public * ArithmeticCalculator.*(..))

含义:ArithmeticCalculator 接口中的所有公有方法

例子3:

表达式:execution(public double ArithmeticCalculator.*(..))

含义:ArithmeticCalculator 接口中返回 double 类型数值的方法

例子4:

表达式:execution(public double ArithmeticCalculator.*(double, ..))

含义:第一个参数为double 类型的方法。.. 匹配任意数量、任意类型的参数。

例子5:

表达式:execution(public double ArithmeticCalculator.*(double, double))

含义:参数类型为double ,double 类型的方法。

(3)在AspectJ中,切入点表达式可以通过 && 或者 || 或者 ! 等操作符结合起来

例子:

表达式:execution(* *.add(int, ..))|| exexution(* *.sub(int, ..))

含义:任意类中第一个参数为 int 类型的 add 方法或 sub 方法

7.2注意事项和细节

  1. 切入表达式可以指向(实现了接口的)类的方法,这时切入表达式会对该类/对象生效

  2. 切入表达式也可以指向接口的方法,这时切入表达式会对实现了接口的类/对象生效

  3. 切入表达式可以对没有实现接口的类进行切入。

    这涉及到CGlib动态代理:动态代理jdk的Proxy与spring的CGlib

  4. 两个动态代理的区别:

    • JDK动态代理是面向接口的,只能增强实现类中接口中存在的方法。CGlib是面向父类的,可以增强父类的所有方法

    • JDK得到的对象是JDK代理对象实例,而CGlib得到的对象是被代理对象的子类

静态代理例子:

一个普通类Car:

package com.li.aop.hw;
import org.springframework.stereotype.Component;
/**
 * @author 李
 * @version 1.0
 */
@Component
public class Car {
    public void run() {
        System.out.println("小汽车在running...");
    }
}

MyAspect切面类:

package com.li.aop.hw;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;
import java.util.Arrays;
/**
 * @author 李
 * @version 1.0
 * 切面类
 */
@Component
@Aspect
public class MyAspect {
    //CGlib
    //给没有实现接口的一个普通类设置前置通知,其他通知亦可以设置
    @Before(value = "execution(public void Car.run())")
    public void ok(JoinPoint joinPoint) {
        String methodName = joinPoint.getSignature().getName();
        System.out.println("MyAspect前置通知ok()-目标方法-" + methodName +
                " 参数-" + Arrays.toString(joinPoint.getArgs()));
    }
}

测试类:

package com.li.aop.hw;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.testng.annotations.Test;
/**
 * @author 李
 * @version 1.0
 */
public class UsbTest {
    @Test
    public void UsbAspectTest() {
        ApplicationContext ioc = new ClassPathXmlApplicationContext("beans08.xml");
        //carProxy是 被代理对象的子类
        Car carProxy = ioc.getBean(Car.class);
        System.out.println("carProxy的实际运行类型=" + carProxy.getClass());
        carProxy.run();
    }
}

测试结果:

day10-AOP-03

8.AOP-JoinPoint

在切面类的通知方法中,可以通过 JoinPoint对象获取到目标方法的一系列信息:

JoinPoint对象的方法 释义
getSignature().getName() 获取目标方法名
getSignature().getDeclaringType().getSimpleName() 获取目标方法所属类的简单类名
getSignature().getDeclaringTypeName() 获取目标方法所属类的全类名
getSignature().getModifiers() 获取目标方法声明类型(public/private/protected)
getArgs() 获取传入目标方法的参数,返回一个数组
getTarget() 获取被代理的对象
getThis() 获取代理对象自己

9.AOP-返回通知获取结果

我们在 @AfterReturning (返回通知)的注解源码中可以看到有一个returning属性。通过 returning 属性可以获取目标方法执行完毕后,返回的结果。

底层大概是:在反射执行目标方法时,将目标方法返回的结果赋给 returning 的定义的变量,然后赋给切入方法同名的参数。

day10-AOP-03

例子

  1. 必须在返回通知中,才能获取目标方法的返回值

  2. returning 属性定义的变量,要和切入方法接收的参数名称一致。

day10-AOP-03

10.AOP-异常通知中获取异常

异常通知 @AfterThrowing 中有一个throwing 的属性,它可以接收异常信息

day10-AOP-03

例子

day10-AOP-03
day10-AOP-03

11.AOP-环绕通知

以AOP-02-6.2 快速入门为例子

接口为 SmartAnimal.java,实现类为 SmartDog.java

切面类为 SmartAnimalAspect2.java:

package com.li.aop.aspectj;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
import java.util.Arrays;
import java.util.List;
/**
 * @author 李
 * @version 1.0
 * 切面类-这里主要演示环绕通知
 */
@Aspect
@Component
public class SmartAnimalAspect2 {
    //环绕通知
    /**
     * 1.@Around 表示环绕通知,它可以完成其他四个通知的功能
     * 2.value = "execution(...)" 切入表达式
     * 3.doAround() 表示要切入的方法,调用结构为 try-catch-finally
     *
     * @param joinPoint 如果是环绕通知,需要用到 ProceedingJoinPoint
     * @return
     */
    @Around(value = "execution(public float com.li.aop.aspectj.SmartDog.getSum(float, float))")
    public Object doAround(ProceedingJoinPoint joinPoint) {
        Object result = null;
        String methodName = joinPoint.getSignature().getName();
        try {
            //1.相当于前置通知完成的事情
            Object[] args = joinPoint.getArgs();
            List<Object> argList = Arrays.asList(args);
            System.out.println("AOP 环绕通知 " + methodName + "方法开始了--参数有:" + argList);
            //在环绕通知中一定要调用 joinPoint.proceed()来执行目标方法
            result = joinPoint.proceed();
            //2.相当于返回通知完成的事情
            System.out.println("AOP 环绕通知 " + methodName + "方法结束了--结果是:" + result);
        } catch (Throwable throwable) {
            //3.相当于异常通知完成的事情
            System.out.println("AOP 环绕通知 " + methodName + "方法抛异常了--异常对象:" + throwable);
        } finally {
            //4.相当于最终通知完成的事情
            System.out.println("AOP 环绕通知 " + methodName + "方法最终结束了...");
        }
        return result;
    }
}

测试方法:

@Test
public void testDoAround(){
    //得到ioc容器
    ApplicationContext ioc = new ClassPathXmlApplicationContext("beans07.xml");
    //获取代理对象
    SmartAnimal smartAnimal = ioc.getBean(SmartAnimal.class);
    //执行方法
    smartAnimal.getSum(99,88);
}

测试结果:

day10-AOP-03

12.AOP-切入点表达式重用

以AOP-02-6.2 快速入门为例子

接口为 SmartAnimal.java,实现类为 SmartDog.java

切面类为 SmartAnimalAspect.java:

package com.li.aop.aspectj;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
import java.util.Arrays;
/**
 * @author 李
 * @version 1.0
 * 切面类
 */
@Aspect 
@Component
public class SmartAnimalAspect {
    //定义一个切入点,在后面使用时可以直接引用,提高复用性
    @Pointcut(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))")
    public void myPointCut() {
    }
    //前置通知
    //@Before(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))")
    //这里我们使用定义好的切入点
    @Before(value = "myPointCut()")
    public void f1(JoinPoint joinPoint) {
        //通过连接点对象joinPoint 拿到方法签名
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类f1()-方法执行开始-日志-方法名-" + signature.getName() +
                "-参数 " + Arrays.toString(joinPoint.getArgs()));
    }
    //返回通知
    //@AfterReturning(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))", returning = "res")
    @AfterReturning(value = "myPointCut()", returning = "res")
    public void f2(JoinPoint joinPoint, Object res) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类f2()-方法执行正常结束-日志-方法名-" + signature.getName());
        System.out.println("目标方法返回的结果=" + res);
    }
    //异常通知
    //@AfterThrowing(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))", throwing = "throwable")
    @AfterThrowing(value = "myPointCut()", throwing = "throwable")
    public void f3(JoinPoint joinPoint, Throwable throwable) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类f3()-方法执行异常-日志-方法名-" + signature.getName() + "异常信息-" + throwable);
    }
    //最终通知
    //@After(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))")
    @After(value = "myPointCut()")
    public void f4(JoinPoint joinPoint) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类f4()-方法最终执行完毕-日志-方法名-" + signature.getName());
    }
}

测试:

@Test
public void test(){
    //得到ioc容器
    ApplicationContext ioc = new ClassPathXmlApplicationContext("beans07.xml");
    //获取代理对象
    SmartAnimal smartAnimal = ioc.getBean(SmartAnimal.class);
    //执行方法
    smartAnimal.getSum(99,88);
}

测试结果:

day10-AOP-03

13.AOP-切面优先级问题

如果同一个方法,有多个切面在同一个切入点切入,那么执行的优先级如何控制?

答:在切面类声明时,使用@Order注解控制优先级:n值越小,优先级越高

@Order(value=n) //n值越小,优先级越高

例子-以AOP-02-6.2 快速入门为例子

接口为 SmartAnimal.java,实现类为 SmartDog.java,两个切面类: SmartAnimalAspect.java 和 SmartAnimalAspect2.java

SmartAnimalAspect.java:

package com.li.aop.aspectj;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.*;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
import java.util.Arrays;
/**
 * @author 李
 * @version 1.0
 * 切面类1
 */
@Aspect 
@Component 
@Order(value = 2)//表示该切面类执行的顺序,value越小,优先级越高
public class SmartAnimalAspect {
    //定义一个切入点,在后面使用时可以直接引用,提高复用性
    @Pointcut(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))")
    public void myPointCut() {
    }
    //前置通知
    @Before(value = "myPointCut()")
    public void f1(JoinPoint joinPoint) {
        //通过连接点对象joinPoint 拿到方法签名
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类1-f1()-方法执行开始-日志-方法名-" + signature.getName() +
                "-参数 " + Arrays.toString(joinPoint.getArgs()));
    }
    //返回通知
    @AfterReturning(value = "myPointCut()", returning = "res")
    public void f2(JoinPoint joinPoint, Object res) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类1-f2()-方法执行正常结束-日志-方法名-" + signature.getName() + " 目标方法返回的结果=" + res);
    }
    //异常通知
    @AfterThrowing(value = "myPointCut()", throwing = "throwable")
    public void f3(JoinPoint joinPoint, Throwable throwable) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类1-f3()-方法执行异常-日志-方法名-" + signature.getName() + "异常信息-" + throwable);
    }
    //最终通知
    @After(value = "myPointCut()")
    public void f4(JoinPoint joinPoint) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类1-f4()-方法最终执行完毕-日志-方法名-" + signature.getName());
    }
}

SmartAnimalAspect2.java:

package com.li.aop.aspectj;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.*;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
import java.util.Arrays;
/**
 * @author 李
 * @version 1.0
 * 切面类 2
 */
@Aspect
@Component
@Order(value = 1)
public class SmartAnimalAspect2 {
    @Before(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))")
    public void f1(JoinPoint joinPoint) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类2-f1()-方法执行开始-日志-方法名-" + signature.getName() +
                "-参数 " + Arrays.toString(joinPoint.getArgs()));
    }
    //返回通知:
    @AfterReturning(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))", returning = "res")
    public void f2(JoinPoint joinPoint, Object res) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类2-f2()-方法执行正常结束-日志-方法名-" + signature.getName() + " 目标方法返回的结果=" + res);
    }
    //异常通知:
    @AfterThrowing(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))", throwing = "throwable")
    public void f3(JoinPoint joinPoint, Throwable throwable) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类2-f3()-方法执行异常-日志-方法名-" + signature.getName() + "异常信息-" + throwable);
    }
    //最终通知:
    @After(value = "execution(public float com.li.aop.aspectj.SmartDog.*(float, float))")
    public void f4(JoinPoint joinPoint) {
        Signature signature = joinPoint.getSignature();
        System.out.println("切面类2-f4()-方法最终执行完毕-日志-方法名-" + signature.getName());
    }
}

测试方法:

@Test
public void smartDogTestByAspectj() {
    //得到Spring容器
    ApplicationContext ioc =
            new ClassPathXmlApplicationContext("beans07.xml");
    //通过接口类型来获得注入的SmartDog对象(实际上是代理对象proxy)
    SmartAnimal smartAnimal = ioc.getBean(SmartAnimal.class);
    smartAnimal.getSum(100, 48);
}

测试结果:

day10-AOP-03

注意事项和细节:

不能理解成:优先级高的切面类,每个消息通知都先执行。

真实的执行顺序和 Filter 过滤器链式调用类似:

day10-AOP-03

14.基于XML配置AOP

前面我们都是通过注解来配置aop的,在spring中,同样支持通过xml的方式来配置aop

例子

1.SmartAnimal 接口:

package com.li.aop.xml;
/**
 * @author 李
 * @version 1.0
 */
public interface SmartAnimal {
    //求和
    float getSum(float a, float b);
    //求差
    float getSub(float a, float b);
}

2.SmartDog 实现类:

package com.li.aop.xml;
/**
 * @author 李
 * @version 1.0
 */
public class SmartDog implements SmartAnimal {
    @Override
    public float getSum(float a, float b) {
        float result = a + b;
        System.out.println("方法内部打印 result = " + result);
        return result;
    }
    @Override
    public float getSub(float a, float b) {
        float result = a - b;
        System.out.println("方法内部打印 result = " + result);
        return result;
    }
}

3.SmartAnimalAspect 切面类:

package com.li.aop.xml;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.Signature;
import java.util.Arrays;
/**
 * @author 李
 * @version 1.0
 * 这是一个切面类,使用xml的方法配置
 */
public class SmartAnimalAspect {
    public void showBeginLog(JoinPoint joinPoint) {
        Signature signature = joinPoint.getSignature();
        System.out.println("xml-showBeginLog()-方法执行开始-日志-方法名-" + signature.getName() +
                "-参数 " + Arrays.toString(joinPoint.getArgs()));
    }
    public void showSuccessEndLog(JoinPoint joinPoint, Object res) {
        Signature signature = joinPoint.getSignature();
        System.out.println("xml-showSuccessEndLog()-方法执行正常结束-日志-方法名-" + signature.getName() + " 目标方法返回的结果=" + res);
    }
    public void showExceptionLog(JoinPoint joinPoint, Throwable throwable) {
        Signature signature = joinPoint.getSignature();
        System.out.println("xml-showExceptionLog()-方法执行异常-日志-方法名-" + signature.getName() + "异常信息-" + throwable);
    }
    public void showFinallyEndLog(JoinPoint joinPoint) {
        Signature signature = joinPoint.getSignature();
        System.out.println("xml-showFinallyEndLog()-方法最终执行完毕-日志-方法名-" + signature.getName());
    }
}

4.xml容器文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/aop
       https://www.springframework.org/schema/aop/spring-aop.xsd">
    <!--使用xml配置完成aop编程-->
    <!--配置一个切面类对象 bean-->
    <bean class="com.li.aop.xml.SmartAnimalAspect" id="smartAnimalAspect"/>
    <!--配置一个SmartDog对象-->
    <bean class="com.li.aop.xml.SmartDog" id="smartDog"/>
    <!--配置切面类 (注意要引入aop名称空间)-->
    <aop:config>
        <!--先配置切入点-->
        <aop:pointcut id="myPointCut" expression="execution(public float com.li.aop.xml.SmartDog.*(float, float))"/>
        <!--配置切面的前置/返回/异常/最终通知-->
        <aop:aspect ref="smartAnimalAspect" order="10">
            <!--配置前置通知-->
            <aop:before method="showBeginLog" pointcut-ref="myPointCut"/>
            <!--配置返回通知-->
            <aop:after-returning method="showSuccessEndLog" pointcut-ref="myPointCut" returning="res"/>
            <!--配置异常通知-->
            <aop:after-throwing method="showExceptionLog" pointcut-ref="myPointCut" throwing="throwable"/>
            <!--最终通知-->
            <aop:after method="showFinallyEndLog" pointcut-ref="myPointCut"/>
            <!--还可以配置环绕通知...-->
        </aop:aspect>
    </aop:config>
</beans>

5.测试类:

package com.li.aop.xml;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.testng.annotations.Test;
/**
 * @author 李
 * @version 1.0
 * 测试类
 */
public class AopAspectjXMLTest {
    @Test
    public void testAspectjByXML() {
        ApplicationContext ioc = new ClassPathXmlApplicationContext("beans09.xml");
        SmartAnimal smartAnimalProxy = ioc.getBean(SmartAnimal.class);
        smartAnimalProxy.getSum(1000, 888);
    }
}

测试结果:

day10-AOP-03

15.aop练习

  1. 请编写一个Cal接口,该接口有两个方法:

    • cal1(int n),计算1 + 2 + ... + n
    • cal2(int n),计算1 * 2 *... * n
  2. Cal 实现类为 MyCal

  3. 请分别使用注解方式和xml配置的方式,完成aop编程

    • 在执行 cal2 前打印开始执行的时间,在执行完后打印时间
    • 在执行 cal2 前打印开始执行的时间,在执行完后打印时间

(1)xml配置的方式:

Cal:

package com.li.aop.hw2;
/**
 * @author 李
 * @version 1.0
 */
public interface Cal {
    //累加
    public void cal1(int n);
    //累乘
    public void cal2(int n);
}

MyCal:

package com.li.aop.hw2;
/**
 * @author 李
 * @version 1.0
 */
public class MyCal implements Cal {
    @Override
    public void cal1(int n) {
        int result = 0;
        if (n >= 1) {
            for (int i = 0; i <= n; i++) {
                result += i;
            }
            System.out.println("cal1-result=" + result);
            return;
        }
        System.out.println("cal1-参数有误");
    }
    @Override
    public void cal2(int n) {
        int result = 1;
        if (n >= 1) {
            for (int i = 1; i <= n; i++) {
                result *= i;
            }
            System.out.println("cal2-result=" + result);
            return;
        }
        System.out.println("cal2-参数有误");
    }
}

切面类:

package com.li.aop.hw2;
/**
 * @author 李
 * @version 1.0
 * 切面类
 */
public class CalAspect {
    //前置通知
    public void beforeTime() {
        System.out.println("开始执行计算 "+System.currentTimeMillis());
    }
    //返回通知
    public void returningTime() {
        System.out.println("结束执行计算 "+System.currentTimeMillis());
    }
}

xml容器配置文件:

<!--配置实现类对象bean-->
<bean class="com.li.aop.hw2.MyCal" id="myCal"/>
<!--配置切面类对象bean-->
<bean class="com.li.aop.hw2.CalAspect" id="calAspect"/>
<aop:config>
    <!--配置切入点表达式-->
    <aop:pointcut id="myPointCut" expression="execution(public void com.li.aop.hw2.MyCal.*(int))"/>
    <!--配置切面类-->
    <aop:aspect ref="calAspect">
        <!--前置通知-->
        <aop:before method="beforeTime" pointcut-ref="myPointCut"/>
        <!--返回通知-->
        <aop:after-returning method="returningTime" pointcut-ref="myPointCut"/>
    </aop:aspect>
</aop:config>

测试方法:

@Test
public void myCalTest() {
    ApplicationContext ioc = new ClassPathXmlApplicationContext("beans10.xml");
    //因为这里只有一个实现对象,因此用类型获取
    Cal myCalProxy = ioc.getBean(Cal.class);
    myCalProxy.cal1(200);
    System.out.println("===============");
    myCalProxy.cal2(5);
}

测试结果:

day10-AOP-03

(2)基于注解的配置方式

Cal 接口不变,在MyCal 实现类上添加 @Component 注解:

@Component
public class MyCal implements Cal {...}

切面类:

package com.li.aop.hw2;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;
/**
 * @author 李
 * @version 1.0
 * 切面类
 */
@Component
@Aspect
public class CalAspect {
    //前置通知
    //如果切面类和目标类在同一个包,可以省略包名
    @Before(value = "execution(public void com.li.aop.hw2.Cal.*(int))")
    public void beforeTime(JoinPoint joinPoint) {
        Signature signature = joinPoint.getSignature();
        System.out.println(signature.getName() + " 开始时间: " + System.currentTimeMillis());
    }
    //返回通知
    @AfterReturning(value = "execution(public void com.li.aop.hw2.Cal.*(int))")
    public void returningTime(JoinPoint joinPoint) {
        Signature signature = joinPoint.getSignature();
        System.out.println(signature.getName() + " 结束时间: " + System.currentTimeMillis());
    }
}

xml容器配置文件:

<!--配置要扫描的包-->
<context:component-scan base-package="com.li.aop.hw2"/>
<!--开启基于注解的aop功能-->
<aop:aspectj-autoproxy/>

测试方法不变:

@Test
public void myCalTest() {
    ApplicationContext ioc = new ClassPathXmlApplicationContext("beans10.xml");
    //因为这里只有一个实现对象,因此用类型获取
    //又因为是代理对象,因此使用接口类型获取
    Cal myCalProxy = ioc.getBean(Cal.class);
    myCalProxy.cal1(200);
    System.out.println("===============");
    myCalProxy.cal2(5);
}

测试结果:

day10-AOP-03

发表回复