Spring Boot是很优秀的框架,它的出现简化了新Spring应用的初始搭建以及开发过程,大大减少了代码量,目前已被大多数企业认可和使用。这个专栏将对Spring Boot框架从浅入深,从实战到进阶,不但我们要懂得如何去使用,还要去剖析框架源码,学习其优秀的设计思想。
汇总目录链接:【Spring Boot实战与进阶】学习目录
文章目录- 一、JDK和Cglib两种动态代理方式区别
- 二、AOP的实现原理
- 1、 引入AOP依赖
- 2、AOP的配置
- 示例一(Cglib动态代理,默认)
- 示例二(JDK动态代理,只能对实现了接口的类生成代理)
- 1、 引入AOP依赖
- 2、AOP的配置
一、JDK和Cglib两种动态代理方式区别
1、java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。而Cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。 2、JDK动态代理只能对实现了接口的类生成代理,而不能针对类;Cglib是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法,因为是继承,所以该类或方法最好不要声明成final 。 3、Cglib一个目标类方法会生成两个代理方法,一个重写目标方法,并实现代理逻辑,还有一个直接调用目标类方法。
二、AOP的实现原理
想了解原理的可以看这篇【Spring教程】详解AOP的实现原理(动态代理),接下来我们直接通过示例讲如何使用。
1、 引入AOP依赖
代码语言:javascript复制<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
2、AOP的配置
2.1 基于Spring的AOP写法
代码语言:javascript复制spring.aop.auto=true # 是否启用aop
spring.aop.proxy-target-class=false # 代理方式有接口使用jdk动态代理,如果没有接口使用cglib代理;设置为true时表示强制使用cglib代理
2.2 基于SpringBoot做法 @EnableAopProxyClass
代码语言:javascript复制exposeProxy属性表示如果使用true就可以使用AopContext对象获取当前代理对象,false则不能使用
proxyTargetClass true表示使用jdk的动态代理, false表示使用cglib代理
示例一(Cglib动态代理,默认)
1、BookDaoImpl
代码语言:javascript复制@Repository
public class BookDaoImpl implements BookDao {
@Override
public void addBook(String name, String author) {
System.out.println("BookDaoImpl实现类:bookName:" name ", bookAuthor:" author);
}
}
2、BookAop
代码语言:javascript复制@Component
@Aspect
public class BookAop {
// 定义切入点
public static final String POINT_CUT = "execution(* com.example.bootaop.dao..*.*(..))";
@Before(POINT_CUT)
public void before() {
System.out.println("----------添加图书方法前[校验]-----------");
}
@After(POINT_CUT)
public void after(JoinPoint jp) {
System.out.println("----------添加图书成功后-----------");
System.out.println(jp.getTarget().getClass());
System.out.println(Arrays.asList(jp.getArgs()));
}
}
3、BootAopApplication
代码语言:javascript复制@SpringBootApplication
public class BootAopApplication {
public static void main(String[] args) {
ConfigurableApplicationContext
context = SpringApplication.run(BootAopApplication.class, args);
BookDaoImpl bookDao = context.getBean(BookDaoImpl.class);
System.out.println(bookDao.getClass());
bookDao.addBook("神雕侠侣", "金庸");
context.close();
}
}
控制台输出
代码语言:javascript复制class com.example.bootaop.dao.BookDaoImpl$$EnhancerBySpringCGLIB$$2d857802
----------添加图书方法前[校验]-----------
BookDaoImpl实现类:bookName:神雕侠侣, bookAuthor:金庸
----------添加图书成功后-----------
class com.example.bootaop.dao.BookDaoImpl
[神雕侠侣, 金庸]
示例二(JDK动态代理,只能对实现了接口的类生成代理)
1、BookDao
代码语言:javascript复制@Repository
public class BookDaoImpl1 implements BookDao {
@Override
public void addBook(String name, String author) {
}
}
2、BookDaoImpl1
代码语言:javascript复制@Repository
public class BookDaoImpl1 implements BookDao {
@Override
public void addBook(String name, String author) {
}
}
3、BootAopApplication
代码语言:javascript复制@SpringBootApplication
public class BootAopApplication {
public static void main(String[] args) {
ConfigurableApplicationContext
context = SpringApplication.run(BootAopApplication.class, args);
BookDao bookDao = context.getBean("bookDaoImpl1", BookDao.class);//多个实现类时
System.out.println(bookDao.getClass());
bookDao.addBook("鹿鼎记", "金庸");
context.close();
}
}
配置文件增加:spring.aop.proxy-target-class=false 控制台输出
代码语言:javascript复制class com.sun.proxy.$Proxy61
----------添加图书方法前[校验]-----------
----------添加图书成功后-----------
class com.example.bootaop.dao.BookDaoImpl1
[鹿鼎记, 金庸]