一、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 {@Overridepublic int add(int a, int b) {return a + b;}@Overridepublic String update(String id) {return id;}
}
- 使用Proxy类创建接口代理对象
①写InvocationHandler的实现类
class UserDaoProxy implements InvocationHandler{//1.把创建的是谁的代理对象,需要传过来。//有参构造private Object obj;public UserDaoProxy(Object obj){this.obj = obj;}@Overridepublic 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.*.*(..))