AOP创建切面

2024-06-23 02:08
文章标签 创建 aop 切面

本文主要是介绍AOP创建切面,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

AOP,就是面向切面编程。
什么是切面呢?增强+切点就是切面。需要向切面里注入一个增强
前面说了增强,这里我们说一说切点,切点就是特定类的特定方法。
Pointcut = ClassFilter + MethodMatcher.
Advisor = Pointcut + Advice
三种切面类型:

  1. 一般切面
  2. 切点切面
  3. 引介切面

Advisor:一般切面,只包含增强,一般不会直接使用
PointcutAdvisor.切点切面,就是比较通用的Advice + Pointcut.
IntroductionAdvisor:引介切面。

切面的实现类

  1. 静态普通方法名匹配切面StaticMethodMatcherPointcutAdvisor.
    匹配所有的目标类
package com.smart.advisor;import java.lang.reflect.Method;import org.springframework.aop.ClassFilter;
import org.springframework.aop.support.StaticMethodMatcherPointcutAdvisor;public class GreetingAdvisor extends StaticMethodMatcherPointcutAdvisor {public boolean matches(Method method, Class clazz) {return "greetTo".equals(method.getName());}   public ClassFilter getClassFilter(){return new ClassFilter(){public boolean matches(Class clazz){return Waiter.class.isAssignableFrom(clazz);}};}
}

只需要实现matches方法,匹配所有的类。
覆盖getClassFilter().让他仅匹配waiter类极其子类。
我们还需要一个增强

package com.smart.advisor;import java.lang.reflect.Method;import org.springframework.aop.MethodBeforeAdvice;public class GreetingBeforeAdvice implements MethodBeforeAdvice {public void before(Method method, Object[] args, Object obj) throws Throwable {String clientName = (String)args[0];System.out.println(obj.getClass().getName()+"."+method.getName());System.out.println("How are you!Mr."+clientName+".");}
}

spring配置来定义切面

    <!-- 普通方法名匹配切面 --><bean id="waiterTarget" class="com.smart.advisor.Waiter" /><bean id="sellerTarget" class="com.smart.advisor.Seller" /><bean id="greetingAdvice"class="com.smart.advisor.GreetingBeforeAdvice" /><!--切面中注入增强--><bean id="greetingAdvisor" class="com.smart.advisor.GreetingAdvisor"p:advice-ref="greetingAdvice" /><!--公共配置信息--><bean id="parent" abstract="true"class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="greetingAdvisor" p:proxyTargetClass="true" /><!--两个代理Bean通过parent放入代理工厂中--><bean id="waiter" parent="parent" p:target-ref="waiterTarget" /><bean id="seller" parent="parent" p:target-ref="sellerTarget" />

当有多个类似的方法,需要配置切面的时候,就应该使用正则表达式方法匹配切面。
RegexpMethodPointcutAdvisor

    <!-- 正则表达式方法名匹配切面 --><bean id="regexpAdvisor"class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"p:advice-ref="greetingAdvice"> <!--增强注入--><property name="patterns"> <!--匹配多个正则方法--><list><value>.*greet.*</value></list></property></bean><bean id="waiter1" class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="regexpAdvisor"  <!--表明切面-->p:target-ref="waiterTarget" <!--表明目标类-->p:proxyTargetClass="true" />

动态切面
应用场景:比如服务员打招呼,只有当顾客是刘洋的时候,她才启用增强。
这就是动态界面,和方法的入参有关系。的切面
通过继承DynamicMethodMathcherPointcut类来实现。这个只是切点类。
通过配置,切点类+增强类 就有了切面。

package com.smart.advisor;import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;import org.springframework.aop.support.DynamicMethodMatcherPointcut;public class GreetingDynamicPointcut extends DynamicMethodMatcherPointcut {private static List<String> specialClientList = new ArrayList<String>();static {specialClientList.add("John");specialClientList.add("Tom");}
//  public ClassFilter getClassFilter() {
//      return new ClassFilter() {
//          public boolean matches(Class clazz) {
//              System.out.println("调用getClassFilter()对"+clazz.getName()+"做静态检查.");
//              return Waiter.class.isAssignableFrom(clazz);
//          }
//      };
//  }
//  public boolean matches(Method method, Class clazz) {
//      System.out.println("调用matches(method,clazz)对"+clazz.getName()+"."+method.getName()+"做静态检查.");
//      return "greetTo".equals(method.getName());
//  }public boolean matches(Method method, Class clazz, Object[] args) {System.out.println("调用matches(method,clazz)对"+clazz.getName()+"."+method.getName()+"做动态检查.");String clientName = (String) args[0];return specialClientList.contains(clientName);}}<!-- 动态切面 --><bean id="dynamicAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"><property name="pointcut"><bean class="com.smart.advisor.GreetingDynamicPointcut" /></property><property name="advice"><bean class="com.smart.advisor.GreetingBeforeAdvice" /></property></bean><bean id="waiter2" class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="dynamicAdvisor" p:target-ref="waiterTarget"p:proxyTargetClass="true" />

流程切面:
前面的动态切面和入参有关,流程切面和调用者有关系。比如:只有通过service调用的greetto,和serveto才能触发他们的增强。通过ControlFlowPointcut实现。

package com.smart.advisor;public class WaiterDelegate {private Waiter waiter;public void service(String clientName) {waiter.greetTo(clientName);waiter.serveTo(clientName);}public void setWaiter(Waiter waiter) {this.waiter = waiter;}
}

通过spring配置切面:

    <!-- 控制流程切面 --><bean id="controlFlowPointcut" class="org.springframework.aop.support.ControlFlowPointcut"><constructor-arg type="java.lang.Class"value="com.smart.advisor.WaiterDelegate" /><constructor-arg type="java.lang.String" value="service" /><!--service 所有的方法都会织入增强--></bean><bean id="controlFlowAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"p:pointcut-ref="controlFlowPointcut" p:advice-ref="greetingAdvice" /><bean id="waiter3" class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="controlFlowAdvisor" p:target-ref="waiterTarget"p:proxyTargetClass="true" />

复合切点切面:
上面的流程切面,service中所有的方法都会织入增强,如果我们只想greeTO()织入增强,我们应该如何做呢?这时候就可以使用复合切点切面:流程切点+方法名切点
ComposablePointcut.提供了切点之前复合运算的功能。

通过Pointcuts工具类,可以完成切点的交集,并集运算。
intersection
union

package com.smart.advisor;
import org.springframework.aop.Pointcut;
import org.springframework.aop.support.ComposablePointcut;
import org.springframework.aop.support.ControlFlowPointcut;
import org.springframework.aop.support.NameMatchMethodPointcut;public class GreetingComposablePointcut {public Pointcut getIntersectionPointcut(){ComposablePointcut cp = new ComposablePointcut(); //复合切点Pointcut pt1 = new ControlFlowPointcut(WaiterDelegate.class,"service");//流程切点NameMatchMethodPointcut pt2 = new NameMatchMethodPointcut();//方法名切点pt2.addMethodName("greetTo");return cp.intersection(pt1).intersection((Pointcut)pt2);    //注意这里的方法}
}
<!-- 复合切点切面 --><bean id="gcp" class="com.smart.advisor.GreetingComposablePointcut" /><bean id="composableAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"p:pointcut="#{gcp.intersectionPointcut}" p:advice-ref="greetingAdvice" /><bean id="waiter4" class="org.springframework.aop.framework.ProxyFactoryBean"p:interceptorNames="composableAdvisor" p:target-ref="waiterTarget"p:proxyTargetClass="true" />

引介切面:
是引介增强的封装器;

另外:
前面我们都通过proxyFactoryBean配置代理。Spring中我们使用了BeanPostProcessor 来完成这项工作。
实现类:
自动创建代理:根据名字找到代理的目标类,根据切面中的信息找到代理的目标类。

    <!-- 通过Bean名称自动创建代理 --><bean
        class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"p:beanNames="*er" p:interceptorNames="greetingAdvice"p:optimize="true"/>
<!--通过Advisor自动创建代理--><bean id="regexpAdvisor"class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"p:patterns=".*greet.*" p:advice-ref="greetingAdvice"  /><bean
        class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator"  p:proxyTargetClass="true" />

注意:目标类的内部方法调用是不会使用代理类处理的。比如serverto()中调用greetTo 的时候不会被增强、

这篇关于AOP创建切面的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1086019

相关文章

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完

Spring创建Bean的八种主要方式详解

《Spring创建Bean的八种主要方式详解》Spring(尤其是SpringBoot)提供了多种方式来让容器创建和管理Bean,@Component、@Configuration+@Bean、@En... 目录引言一、Spring 创建 Bean 的 8 种主要方式1. @Component 及其衍生注解

AOP编程的基本概念与idea编辑器的配合体验过程

《AOP编程的基本概念与idea编辑器的配合体验过程》文章简要介绍了AOP基础概念,包括Before/Around通知、PointCut切入点、Advice通知体、JoinPoint连接点等,说明它们... 目录BeforeAroundAdvise — 通知PointCut — 切入点Acpect — 切面

MySQL 数据库表操作完全指南:创建、读取、更新与删除实战

《MySQL数据库表操作完全指南:创建、读取、更新与删除实战》本文系统讲解MySQL表的增删查改(CURD)操作,涵盖创建、更新、查询、删除及插入查询结果,也是贯穿各类项目开发全流程的基础数据交互原... 目录mysql系列前言一、Create(创建)并插入数据1.1 单行数据 + 全列插入1.2 多行数据

MySQL 临时表创建与使用详细说明

《MySQL临时表创建与使用详细说明》MySQL临时表是存储在内存或磁盘的临时数据表,会话结束时自动销毁,适合存储中间计算结果或临时数据集,其名称以#开头(如#TempTable),本文给大家介绍M... 目录mysql 临时表详细说明1.定义2.核心特性3.创建与使用4.典型应用场景5.生命周期管理6.注

MySQL的触发器全解析(创建、查看触发器)

《MySQL的触发器全解析(创建、查看触发器)》MySQL触发器是与表关联的存储程序,当INSERT/UPDATE/DELETE事件发生时自动执行,用于维护数据一致性、日志记录和校验,优点包括自动执行... 目录触发器的概念:创建触www.chinasem.cn发器:查看触发器:查看当前数据库的所有触发器的定

创建springBoot模块没有目录结构的解决方案

《创建springBoot模块没有目录结构的解决方案》2023版IntelliJIDEA创建模块时可能出现目录结构识别错误,导致文件显示异常,解决方法为选择模块后点击确认,重新校准项目结构设置,确保源... 目录创建spChina编程ringBoot模块没有目录结构解决方案总结创建springBoot模块没有目录

IntelliJ IDEA2025创建SpringBoot项目的实现步骤

《IntelliJIDEA2025创建SpringBoot项目的实现步骤》本文主要介绍了IntelliJIDEA2025创建SpringBoot项目的实现步骤,文中通过示例代码介绍的非常详细,对大家... 目录一、创建 Spring Boot 项目1. 新建项目2. 基础配置3. 选择依赖4. 生成项目5.

Linux线程之线程的创建、属性、回收、退出、取消方式

《Linux线程之线程的创建、属性、回收、退出、取消方式》文章总结了线程管理核心知识:线程号唯一、创建方式、属性设置(如分离状态与栈大小)、回收机制(join/detach)、退出方法(返回/pthr... 目录1. 线程号2. 线程的创建3. 线程属性4. 线程的回收5. 线程的退出6. 线程的取消7.

创建Java keystore文件的完整指南及详细步骤

《创建Javakeystore文件的完整指南及详细步骤》本文详解Java中keystore的创建与配置,涵盖私钥管理、自签名与CA证书生成、SSL/TLS应用,强调安全存储及验证机制,确保通信加密和... 目录1. 秘密键(私钥)的理解与管理私钥的定义与重要性私钥的管理策略私钥的生成与存储2. 证书的创建与