面对切面编程
目的:降低各功能模块之间的耦合度
通俗描述:不通过修改源代码方式,在主干功能里面添加新功能
一、JDK动态代理
jdk动态处理前提是有接口类。创建接口实现类处理对象
1.调用newProxyInstance方法
其中三个参数分别为:类加载器(当前类的加载器,当前类名.class.getClassLoader() )、增强方法所在的类实现的接口(就是前提中已有的接口)、实现这个接口 InvocationHandler(创建代理对象,写增强的部分)
public class JDKProxy {
public static void main(String[] args) {
Class[] interfaces = {UserDao1.class};
UserDaoImpl userDao = new UserDaoImpl();
UserDao1 dao = (UserDao1) newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new fangfa(userDao));
UserDao1是现有的接口,UserDaoImpl是接口的实现类,fangfa是写增强部分的类名称
2.创建代理对象代码
invoke方法中是增强部分,即除了执行原有的方法外,还可以在invoke方法内添加想要的功能
invoke三个参数:原有的接口的实现类、要执行的方法(也是原有的实现类中的方法)、要执行的方法的参数。
//创建代理对象
class fangfa implements InvocationHandler {
//有参构造
private Object obj;
public fangfa(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);
//方法之后
if (method.getName() == "update") {
System.out.println("haha");
}
return res;
}
}
二、基于 AspectJ 实现 AOP 操作
1、AOP准备工作
1.引入依赖
2.切入点表达式
切入点表达式作用:知道对哪个类里面的哪个方法进行增强
语法结构: execution([权限修饰符] [返回类型(可不写)] [类全路径] [方法名称]([参数列表]) )
2. AspectJ 注解(重点)
1.在xml文件中,开启注解扫描。增强类和被增强类都要被注解
(注解部分可以参考Spring IOC部分梳理(B)_小曹来巡山的博客-优快云博客)
2.在增强类上要在加上Aspect注解(表示生成代理对象),然后在配置文件中配置Aspect,注意要开辟aop名称空间。
<!-- 开启Aspect生成代理对象-->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
3.在增强类里面,在作为通知(增强部分)方法上面添加通知类型注解,使用切入点表达式配置
通知类型(Before 前置通知、AfterReturning后置通知、After最终通知、Around环绕通知、AfterThrowing异常通知)
后置通知
//后置通知(返回通知)
@AfterReturning(value = "execution(* com.atguigu.spring5.aopanno.User.add(..))")
public void afterReturning() {
System.out.println("afterReturning.........");
}
环绕通知
//环绕通知
@Around(value = "execution(* com.atguigu.spring5.aopanno.User.add(..))")
public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
System.out.println("环绕之前.........");
//被增强的方法执行
proceedingJoinPoint.proceed();
System.out.println("环绕之后.........");
}
引入相同的切入点
在增强类中创建一个空方法,前面加上@Pointcut(value= “execution(~~~)”),后面的增强方法value值为创建的方法名。
//相同切入点抽取
@Pointcut(value = "execution(* com.atguigu.spring5.aopanno.User.add(..))")
public void pointdemo() {
}
//前置通知
//@Before注解表示作为前置通知
@Before(value = "pointdemo()")
public void before() {
System.out.println("before.........");
}
有多个增强类,设置优先级
在增强类上面添加注解@Order(数字),数字表示优先级
完全使用注解开发
3.AspectJ 配置文件(了解)
配置文件和两个类都没有注解
在xml配置文件中创建两个类对象(增强类和被增强类),(通过两个bean标签创建)
在配置文件中配置文件中切入点
<!--创建对象-->
<bean id="book" class="com.atguigu.spring5.aopxml.Book"></bean>
<bean id="bookProxy" class="com.atguigu.spring5.aopxml.BookProxy"></bean>
<!--配置aop增强-->
<aop:config>
<!--切入点,即需要增强的方法-->
<aop:pointcut id="p" expression="execution(* com.atguigu.spring5.aopxml.Book.buy(..))"/>
<!--配置切面-->
<aop:aspect ref="bookProxy"> <!--bookProxy是增强类-->
<!--增强作用在具体的方法上-->
<aop:before method="before" pointcut-ref="p"/> <!--before是增强方法-->
</aop:aspect>
</aop:config>