AOP基本概念
1、什么是AOP
- 面向切面编程(方面),利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度减低,提高程序的可重用性,同时提高了开发的效率。
- 理解为装饰器,和代码关系依赖关系低,即耦合性低
- 通俗解释:不通过修改源代码方式,在主干功能里面添加一个新功能。
AOP底层原理
1、AOP底层使用动态代理
- 有两种情况动态代理
第一种,有接口情况,使用JDK动态代理(创建接口实现类代理对象,增强类的方法)
第二种,没有接口情况,使用CGLIB动态代理(使用子类的动态代理对象,增强类的方法)
JDK动态代理
1、使用JDK动态代理,使用Proxy类里面的方法创建代理对象。
2、调用newProxyInstance方法
方法有三个参数:
- 第一参数,类加载器
- 第二参数,增强方法所在的类,这个类实现的接口,支持多个接口。
- 第三参数,实现这个接口InvocationHandler,创建代理对象,写增强的部分。
代码演示
1、创建接口,且实现接口方法
public interface UserDao {
public int add(int a,int b);
public String update(String id);
}
public class UserDaoImpl implements UserDao{
@Override
public int add(int a, int b) {
return a+b;
}
@Override
public String update(String id) {
return id;
}
}
2、使用Proxy类创建接口代理对象
//JDK代理的底层原理
public class JDKProxy {
public static void main(String[] args) {
UserDaoImpl userDao = new UserDaoImpl();
// 获得代理对象
// 第一参数,类加载器
// 第二参数,增强方法所在的类,这个类实现的接口,支持多个接口。
// 第三参数,实现这个接口InvocationHandler,创建代理对象,写增强的部分。
UserDao dao = (UserDao)Proxy.newProxyInstance(JDKProxy.class.getClassLoader(),interfaces,new UserDaoProxy(userDao));
//执行什么方法,则增强其对应的方法的功能
dao.update("123");//执行update方法
//int result = dao.add(1,3);//执行add方法
//System.out.println("结果:"+result);
}
}
//创建代理对象代码
class UserDaoProxy implements InvocationHandler{
//1 把创建的是谁的代理对象,把谁传递过来
//有参数构造传递
private Object obj;
public UserDaoProxy(Object obj){
this.obj=obj;
}
//增强的功能,即逻辑 method就是方法
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//方法之前,method.getName()得到当前执行的方法
System.out.println("方法之前执行-----"+method.getName()+"方法执行了 传递的参数"+ Arrays.toString(args));
//被增强的方法执行,res即为已经增强后的类,需返回
Object res = method.invoke(obj, args);
//方法之后
System.out.println("方法之后执行-----" + obj);
return res;
}
}
AOP术语
1、连接点:可被增强的方法,即为连接点。
2、切入点:真正实现的连接点。
3、通知(增强):增强的那部分逻辑(代码段)。
4、切面:把通知(增强)应用到切点的过程。
AOP操作(准备)
1、Spring框架一般都是基于AspectJ实现AOP操作
什么是AspectJ?
答:AspectJ不是Spring组成部分,独立AOP框架,一般吧AspectJ和Spring框架一起使用,进行AOP操作。
2、基于AspectJ实现AOP操作
- 基于xml配置文件实现
- 基于注解方式实现(更便捷,常用)
3、在项目工程里面引入AOP相关依赖
4、切入点表达式
1.切入点表达式作用:知道那个类里面的哪个方法进行增强。
2.语法结构:
execution([权限修饰符][返回类型][类全路径][方法名称]([参数列表]))
例子1:对com.atguigu.dao.BookDao类里面的add方法进行增强。
execution(* com.atguigu.dao.BookDao.add(.....))// *星号表示任意修饰符,忽略返回类型
例子2:对com.atguigu.dao.BookDao类里面的所有的方法进行增强。
execution(* com.atguigu.dao.BookDao.*)// *星号表示任意
例子3:对com.atguigu.dao类里面的所有包,所有类里面的所有方法进行增强。
execution(* com.atguigu.dao*)// *星号表示任意