Java架构师交流群:793825326
java版本:jdk1.8
IDE:Idea2019
Springboot:2.1.6.RELEASE
1.pom:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
2.被代理的类:
@Component
public class AopTest {
public void save()
{
System.out.println("hello World");
}
}
3.切面类:
@Component
@Aspect
public class AspectTest {
@Pointcut("execution(public * com.example.demo.AopTest.save())")
public void start()
{}
@After(value = "start()")
private void doAfter()
{
System.out.println("after...");
}
@Before(value = "start()")
private void doBefore()
{
System.out.println("before...");
}
}
表示针对com.example.demo.AopTest.save()进行切面处理。
4.添加一个初始化类Initialize作为测试用:
@Component
@Order
public class Initialize implements CommandLineRunner {
@Autowired
private AopTest aopTest;
@Override
public void run(String... args) throws Exception {
aopTest.save();
}
}
5.启动运行后,打印结果如下:
before...
hello World
after...
6.简单做一下修改,将被代理的类用final修饰:
@Component
public final class AopTest {
public void save()
{
System.out.println("hello World");
}
}
7.编译后运行,报如下异常:
提示我们无法继承final类。同时可以看到是在cglig里面报的异常,这说明springboot是使用cglib来做的代理。由于cglib的代理机制是生成一个代理类,继承被代理对象,所以当被代理对象为final时,就会报异常。
8.如果我们仍然想对final类进行代理该怎么做呢,我们可以尝试使用AspectJ进行代理,将切面类的@Component注解去掉:
@Aspect
public class AspectTest {
@Pointcut("execution(public * com.example.demo.AopTest.save())")
public void start()
{
}
@After(value = "start()")
public void doAfter()
{
System.out.println("after...");
}
@Before(value = "start()")
public void doBefore()
{
System.out.println("before...");
}
}
将编译器由默认的javac改为Ajc,即AspectJ的编译器,如果不了解这块,可以参考我的另外一篇博客:https://blog.youkuaiyun.com/dap769815768/article/details/94019129
编译后启动代码,运行结果如下:
before...
hello World
after...
之所以去掉@Component注解是为了不让SpringBoot去处理切面类,而是交给AspectJ自己的编译器去处理。这样就避免了使用Cglib来做代理了。
实际上SpringBoot处理Aop用的代理方式主要有两种,Java自带的代理和cglib代理,但它同时也兼容AspectJ代理。