一、AOP概念
AOP是指面向切面编程,利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
通俗描述:不通过修改源代码方式,在主干功能里面添加新功能 。
示例:
上例是一个基本的登录功能,我们在完成其基本流程的代码之后,如果要添加一个权限判断的功能,那么传统的做法就只能修改源代码,添加if判断,判断登陆者权限。但是,采用AOP,我们可以不修改源代码,添加新的功能。我们单独编写独立的权限判断模块,并通过配置,将其配置到登录流程中。
二、AOP底层原理
AOP底层使用了动态代理,分为两种情况的动态代理。
2.1 有接口情况
有接口的情况,使用JDK动态代理。
- 创建接口实现类的代理对象,增强类的方法。
如上图所示,有一个接口以及其的一个实现类,用于实现登录功能,我们的目的是不修改源代码,为登录功能添加权限验证,也就是扩展登录功能函数login()。其底层的实现原理是,通过JDK动态代理,创建UserDao接口实现类的代理对象,通过代理对象来增强、扩展原功能(登录功能)。
2.2 没有接口情况
没有接口的情况,使用CGLIB动态代理。
- 创建子类的代理对象,增强类的方法。
示例:
上例中,User类中有一个add方法,要在不修改源代码的前提下,对该方进行增强,由于其没有接口,所以不能采用上节创建接口实现类的代理对象的方式来增强,其采用的底层原理是创建当前类的子类的代理对象,在其子类中,对方法功能进行增强。
2.3 JDK有接口动态代理代码示例:
使用JDK动态代理,需要使用Proxy类里面的方法,来创建代理对象。Proxy
提供了创建动态代理类和实例的静态方法,它也是由这些方法创建的所有动态代理类的超类。方法如下:
方法有三个参数:
- 第一参数,类加载器
- 第二参数,增强方法所在的类,这个类实现的接口,支持多个接口
- 第三参数,实现这个接口 InvocationHandler,创建代理对象,写增强的部分
JDK动态代理代码:
(1)创建接口,定义方法
/**
* 接口
*/
public interface UserDao {
public int add(int a,int b);
public String upData(String id);
}
(2)创建接口实现类,实现方法
public class UserDaoImpl implements UserDao{
@Override
public int add(int a, int b) {
System.out.println("add方法被执行...");
return a+b;
}
@Override
public String upData(String id) {
System.out.println("upData方法被执行...");
return id;
}
}
(3)使用Proxy类创建接口代理对象(增强原功能)
public class JDKProxy {
public static void main(String[] args) {
//创建接口实现类的代理对象
Class[] interfaces = {UserDao.class};
//匿名类的写法
// Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new InvocationHandler() {
// @Override
// public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// return null;
// }
// });
UserDaoImpl userDao = new UserDaoImpl();
UserDao dao = (UserDao) Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(userDao));
int result = dao.add(1,2);
System.out.println("result:"+result);
}
}
//创建代理对象代码
class UserDaoProxy implements InvocationHandler{
//1. 把创建的是谁的代理对象,将其传递过来
//有参构造函数传递
private Object obj;
public UserDaoProxy(Object obj){
this.obj = obj;
}
//增强的逻辑
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//方法之前的增强逻辑
System.out.println("方法之前执行的增强逻辑"+method.getName()+" :传递参数..."+ Arrays.toString(args));
//被增强的方法执行
Object res = method.invoke(obj,args);
//方法之后的增强逻辑
System.out.println("方法之后的增强逻辑"+obj);
return res;
}
}
代码执行结果:
方法之前执行的增强逻辑add :传递参数...[1, 2]
add方法被执行...
方法之后的增强逻辑com.wyf.JDKDemo.UserDaoImpl@31cefde0
result:3
三、结束
本文主要介绍了AOP的基本概念,以及其实现的底层基本原理。