Spring中AOP的理解以及如何用代码实现

Spring中AOP的理解以及如何用代码实现


一、什么是AOP?

AOP(Aspect-Oriented Programming,面向切面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。当我们需 要为分散的对象引入公共行为的时候,OOP则显得无能为力。也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系例如日志功能。日 志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种 散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。

而AOP技术则恰恰相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为 “Aspect”,即方面。所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低 模块间的耦合度,并有利于未来的可操作性和可维护性。AOP代表的是一个横向的关系,如果说“对象”是一个空心的圆柱体,其中封装的是对象的属性和行为; 那么面向方面编程的方法,就仿佛一把利刃,将这些空心圆柱体剖开,以获得其内部的消息。而剖开的切面,也就是所谓的“方面”了。然后它又以巧夺天功的妙手 将这些剖开的切面复原,不留痕迹。

使用“横切”技术,AOP把软件系统分为两个部分:核心关注点和横切关注点。业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点。横 切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处都基本相似。比如权限认证、日志、事务处理。Aop 的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。正如Avanade公司的高级方案构架师Adam Magee所说,AOP的核心思想就是“将应用程序中的商业逻辑同对其提供支持的通用服务进行分离。”

实现AOP的技术是采用动态代理技术,利用截取消息的方式,对该消息进行装饰,以取代原有对象行为的执行;



二、Spring中如何实现AOP

           通过上面的一个总结我们要熟悉target(ps:看成明星)、advice(ps:经纪人)、proxy(ps:艺人--接口),通过配置这三个属性我们就可以实现一个动态代理。


代码实现:

    接口:

       package com.zking.proxyspring;

       public interface ArtorSpring {

public void playcode();
public void sleep();
    }  

    

   对象目标 :

        package com.zking.proxyspring;

       import org.aopalliance.intercept.MethodInterceptor;
       import org.aopalliance.intercept.MethodInvocation;

       public class TagetP implements ArtorSpring{

public void playcode() {
// TODO Auto-generated method stub
System.out.println("代码进行中。。。。。");
}

  }


  通知:

    package com.zking.proxyspring;

    import org.aopalliance.intercept.MethodInterceptor;
    import org.aopalliance.intercept.MethodInvocation;

    public class AdviceY implements MethodInterceptor{

    public Object invoke(MethodInvocation arg0) throws Throwable {
// TODO Auto-generated method stub
Object o=arg0.proceed();
return o;
}

     }   



配置applicationcontext.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">


   <!-- 配置Spring中的动态代理 --> 
    <!-- 配置target:目标 -->
    <bean id="tagetp" class="com.zking.proxyspring.TagetP"></bean>
    
    
    <!-- 配置advice:通知  环绕通知-->
    <bean id="advicey" class="com.zking.proxyspring.AdviceY"></bean>
    
    <!-- 配置前置通知 -->
    <!--<bean id="newadvice" class="com.zking.proxyspring.NewAdvice"></bean>-->
    
    <!-- 配置前置通知过滤器 -->
    <!-- <bean id="mybefore" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
      <property name="advice" ref="advicey"></property>
      <property name="pattern" value=".*play.*"></property>
    </bean>-->
    
    
    <!-- 配置后置通知 -->
    <bean id="afteradvice" class="com.zking.proxyspring.AfterAdvice"></bean>
    
    
    
    <!-- 配置混合代理对象 -->
    <bean id="myproxy" class="org.springframework.aop.framework.ProxyFactoryBean">
     <!-- 引用目标 -->
      <property name="target" ref="tagetp"></property>
     
     <!-- 目标实现所有的接口 --> 
      <property name="proxyInterfaces">
       <list>
         <value>com.zking.proxyspring.ArtorSpring</value>
       </list>
      </property>
     
      <!-- 引用通知 -->
       <property name="interceptorNames">
           <list>   
            <idref bean="advicey"/>
                    
           </list>
       </property>         
    </bean>
</beans>


测试类:

@Test
public void stest(){
ApplicationContext ac=new ClassPathXmlApplicationContext("applicationContext.xml");
ArtorSpring ao=(ArtorSpring) ac.getBean("myproxy");
ao.sleep();

}



总结:AOP从总的来说就是一个面向切面,细一点就是动态代理,在细一点它就是一个通知,通知在什么时候用呢?一般我们用到事物的时候也就是所说的通知了。


ps:通知

1、通知的分类
<1>环绕通知(方法的拦截):MethodInterceptor
<2>前置通知:
实现步骤:
①.通知类实现MethodBeforeAdvice接口
②.重写方法before()
③.配置前置通知类
④.拦截器添加
<property name="interceptorNames">
<list>
<idref bean="前置通知id">
<idref bean="环绕通知id">
</list>
</property>


<3>后置通知:
实现步骤:
①.通知类实现AfterReturningAdvice接口
②.重写方法afterReturning
③.配置后置通知类
④.拦截器添加
<property name="interceptorNames">
<list>
<idref bean="后置通知id">
<idref bean="环绕通知id">
</list>
</property>


注意:配置通知顺序
前置通知--->后置通知--->环绕通知


2、过滤通知
步骤:
①.配置:通知过滤条件
<bean id="通知过滤id" class="RegexpMethodPointcutAdvisor类全限定名">


<property name="advice" ref="通知分类id"></property>
<property name="pattern">
<value>.*匹配字符串.*</value><!-- 配置不过滤的方法 这里方法名可只写前几个字母 -->
</property>
</bean>



  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
AOP(面向切面编程)是Spring框架的一个重要特性,它通过将通用的横切关注点(如日志、事务、安全等)与业务逻辑相分离,从而提高系统的可维护性和可扩展性。 在SpringAOP实现通常通过代理模式来实现,即在调用目标方法前后,通过代理类来拦截并执行额外的逻辑。Spring提供了多种AOP实现方式,包括基于XML配置、基于注解的声明式AOP等。 下面是一个基于注解的AOP示例: 首先定义一个切面类,用于定义切入点和切面逻辑: ```java @Aspect @Component public class LoggingAspect { @Pointcut("execution(* com.example.demo.service.*.*(..))") public void serviceMethod() {} @Before("serviceMethod()") public void logBefore(JoinPoint joinPoint) { System.out.println("Before " + joinPoint.getSignature().getName()); } @After("serviceMethod()") public void logAfter(JoinPoint joinPoint) { System.out.println("After " + joinPoint.getSignature().getName()); } } ``` 上述代码使用了@Aspect注解来声明一个切面类,并定义了一个切入点serviceMethod(),它匹配所有com.example.demo.service包下的方法。在切面类,我们使用@Before和@After注解分别在目标方法执行前和执行后打印日志。 接下来定义一个服务类: ```java @Service public class UserService { public void addUser(String name) { System.out.println("Add user: " + name); } public void deleteUser(String name) { System.out.println("Delete user: " + name); } } ``` 最后,在Spring配置文件声明切面和服务类的bean: ```xml <bean id="loggingAspect" class="com.example.demo.aspect.LoggingAspect"/> <bean id="userService" class="com.example.demo.service.UserService"/> ``` 现在我们可以通过调用UserService的方法来触发切面逻辑: ```java ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userService = context.getBean("userService", UserService.class); userService.addUser("Alice"); userService.deleteUser("Bob"); ``` 输出结果如下: ``` Before addUser Add user: Alice After addUser Before deleteUser Delete user: Bob After deleteUser ``` 可以看到,在调用addUser和deleteUser方法前后,切面逻辑都被正确地执行了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值