AOP中基本概念
-
面向切面编程(方面),利用aop可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各个部分之间的耦合度降低,提高代码的可重用性,同时提高了开发效率。
-
通俗描述:不通过修改源代码方式,在主干功能里面添加新功能
AOP底层原理
-
AOP底层使用动态代理
有两种情况的动态代理
第一种,有接口的情况,使用JDK动态代理
第二种,没有接口的情况,使用CGLIB动态代理
AOP(JDK动态代理实现)
- 使用JDK动态代理,使用Proxy类里面的方法创建代理类
- 调用 java.lang.reflect.Proxy中的newProxyInstance方法
方法中有三个参数
- 1 ClassLoader 类加载器
- 2 增强方法所在的类,这个类实现的接口,支持多个接口
- 3 要实现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) {
System.out.println("执行方法");
return a+b;
}
@Override
public String update(String id) {
return id;
}
}
第三步创建一个类实现InvocationHandler接口
//创建代理对象代码
class UserDaoProxy implements InvocationHandler{
//把被代理对象传递过来 有参构造传递
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 invoke = method.invoke(obj, args);
//
System.out.println("方法之前执行。。。。。。");
return invoke;
//这里返回被代理对象
}
}
第四步调用newProxyInstance()方法
public class JDKProxy {
public static void main(String[] args) {
//创建接口实现类代理对象
Class[] interfaces = {UserDao.class};
UserDaoImpl userDao = new UserDaoImpl();
//这里为什么传JDKProxy.class.getClassLoader()
//百度了一下表示 用哪个类加载器去加载代理对象
//还是不是很懂
UserDao dao = (UserDao) Proxy.newProxyInstance(JDKProxy.class.getClassLoader(),interfaces,new UserDaoProxy(userDao));
int add = dao.add(1, 2);
System.out.println(add);
}
}
AOP术语
-
连接点: 类中可以被增强的方法就叫连接点
-
切入点:实际被真正增强的方法就叫切入点
-
通知(增强) 实际增强的逻辑部分称为通知,方法前后补充的逻辑
前置通知 后置通知 环绕通知 异常通知 最终通知
-
切面 是动作,把通知应用到切入点的过程。
AOP操作
Spring框架一般基于AspectJ实现AOP操作。
- 第一步导入AspectJ依赖
- 第二步创建要被代理的类
/*
* 创建一个类,在类中定义一个方法*/
@Component(value = "zzz")
public class User {
public void add(){
System.out.println("add........");
}
}
- 第三步
/*创建增强类,创建方法,让不同方法代表不同通知类型*/
@Component
@Aspect
public class UserProxy {
//在增强类中的通知方法上面添加通知类型注解,使用切点表达式配置
@Before(value = "execution(* com.csy.aopst.User.add(..))")
public void bofore(){
System.out.println("before.....");
}
}
- 第四步
<!--开启扫描
1如果扫描多个包多个包使用逗号隔开
2扫描包上层目录
-->
<context:component-scan base-package="com.csy.aopst"></context:component-scan>
<!--开启aspectj生产自动代理对象-->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>