一、AOP基本概念
1.什么是AOP
- 面向切面编程(方面),利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
- 不通过修改源代码方式添加新的功能
- 通过画图理解AOP
二、AOP(底层原理)
1.AOP底层使用动态代理
有两种情况的动态代理
- 第一种 有接口的情况,使用JDK动态代理
- 创建接口实现类的代理对象,增强类的方法
- 第二种 没有接口的情况,使用CGLIB动态代理
- 创建子类的代理对象,增强类的方法
三、AOP(JDK动态代理)
1.使用JDK动态代理,使用Proxy类里面的方法创建代理对象
- 调用newProxyInstance方法
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
①参数一ClassLoader类加载器
②参数二Class<?>[] 增强方法所在的类,这个类实现的接口,支持多个接口
③参数三InvocationHandler实现这个接口,创建代理对象,实现增强的方法。
2.JDK动态代理代码
- 创建接口,定义方法
public interface UseDao {
public int add(int a,int b);
public String update(String id);
}
- 创建接口的实现类,实现接口的方法
public class UserDaoImpl implements UseDao {
@Override
public int add(int a, int b) {
return a + b;
}
@Override
public String update(String id) {
return id;
}
}
- 使用Proxy类创建接口代理对象
①写InvocationHandler的实现类
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.toString());
return res;
}
}
②写创建接口实现类的代理对象
public static void main(String[] args) {
//创建接口实现类的代理对象
Class[] interfaces = {UseDao.class};
UseDao dao = (UseDao)Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(new UserDaoImpl()));
// 当代理对象调用真实对象的方法时,自动跳转到代理对象关联handler的invoke方法调用
int result = dao.add(1, 2);
String ddd = dao.update("ddd");
}
结果是
四、AOP(术语)
1.连接点
类里面哪些方法可以被增强,这些方法成为连接点。如:add(),update()…
2.切入点
实际被真正增强的方法,成为切入点。如只增强“add()”方法
3.通知(增强)
- 实际增强的逻辑部分成为通知
- 通知有多种类型:如add方法
*前置通知:add方法之前执行
*后置通知:add方法之后执行
*环绕通知:在add方法的前面和后面都执行
*异常通知:当add方法出现异常执行
*最终通知:finally
4.切面
是动作
- 把通知应用到切入点的过程
五、AOP操作(准备)
1.Spring框架一般基于AspectJ实现AOP操作
- 什么是AspectJ
*AspectJ不是Spring组成部分,独立AOP框架,一般把AspectJ和Spring框架一起使用,进行AOP操作
2.基于AspectJ实现AOP操作
- 基于xml配置文件实现
- 基于注解方式实现(使用)
3.在项目工程里面引入AOP相关依赖
4.切入点表达式
- 切入点表达式作用:知道对哪个类里面的哪个方法进行增强
- 语法结构
execution([权限修饰符][返回类型][类全路径][方法名称]([参数列表]))
举例1:对com.zhilei.dao.BookDao类里面的add进行增强
execution(*com.dao.BookDao.add(..))修饰符可以省略,返回类型用*,代表参数..
举例2:对com.zhilei.dao.BookDao类里面的所有方法进行增强
execution(*com.dao.BookDao.*(..))
举例3:对com.zhilei.dao包里的所有类,所有方法,进行增强
execution(*com.dao.*.*(..))