草庐IT

Spring(十四):SpringAOP及AOP的三种实现方法

jmsstudy 2023-04-17 原文

一、什么是AOP

AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

二、AOP的一些概念

1.Aspect(切面):切面是通知和切入点的结合。

2.Join point(连接点):与切入点匹配的执行点,例如执行方法或处理异常。在SpringAOP 中,连接点始终表示方法执行。

3.Advice(通知):在切面中需要完成的工作。

4.Pointcut(切入点):切入通知执行的位置。

5.Target(目标):被通知的对象。

6. proxy(代理):向目标对象应用通知之后创建的对象。

7.Introduction(引入):向现有类添加新的方法或属性。

8.Weaving(织入):将各个方面与其他应用程序类型或对象链接起来,以创建通知的对象。

三、AOP的三种实现方法

首先我们需要导入依赖:

     <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.9.9.1</version>
        </dependency>

1.通过SpringAPI接口进行实现

SpringAOP有五种通知方式,也有对应的接口:

Before:前置通知,在目标方法调用前通知,对应接口:MethodBeforeAdvice;

After:后置通知,在目标方法返回或异常后通知,对应接口:AfterReturningAdvice;

AfterReturning:后置返回通知,在目标方法返回后通知,对应接口:AfterReturningAdvice;

AfterThrowing:异常通知,在目标方法抛出异常后通知,对应接口:ThrowsAdvice;

Around:环绕通知:通知方法会将目标方法包裹起来,对应接口:MethodInterceptor;

我们下面以具体的例子来展示:

(1)定义一个接口

package com.jms.service;

public interface UserService {
    void create();
    void read();
    void update();
    void delete();
}

(2)接口的实现类

package com.jms.service;

public class UserServiceImpl implements UserService{
    @Override
    public void create() {
        System.out.println("建立了一个用户信息");
    }

    @Override
    public void read() {
        System.out.println("读取了一个用户信息");
    }

    @Override
    public void update() {
        System.out.println("更新了一个用户信息");
    }

    @Override
    public void delete() {
        System.out.println("删除了了一个用户信息");
    }
}

(3)建立两个类分别继承前置通知和后置返回通知的接口

package com.jms.log;

import org.springframework.aop.MethodBeforeAdvice;

import java.lang.reflect.Method;

public class beforeLog implements MethodBeforeAdvice {
    @Override
    //method:要执行目标对象的方法
    //args:参数
    //target:目标对象
    public void before(Method method, Object[] args, Object target) throws Throwable {
        System.out.println("[Debug]" + target.getClass().getName() + "的" + method.getName() + "执行...");
    }
}
package com.jms.log;

import org.springframework.aop.AfterReturningAdvice;

import java.lang.reflect.Method;

public class afterLog implements AfterReturningAdvice{
    //returnValue:返回值
    @Override
    public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
        System.out.println("[Debug]" + target.getClass().getName() + "的" + method.getName() + "执行完成,返回了" + returnValue);
    }
}

(4)xml配置文件

注册bean,并且配置aop

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

    <bean id="userService" class="com.jms.service.UserServiceImpl"/>
    <bean id="beforeLog" class="com.jms.log.beforeLog"/>
    <bean id="afterLog" class="com.jms.log.afterLog"/>

    <!--方法一:使用SpringAPI接口-->
    <!--aop配置-->
    <aop:config>
        <!--pointcut:切入点
            expression:表达式
        -->
        <aop:pointcut id="pointcut" expression="execution(* com.jms.service.UserServiceImpl.*(..))"/>
        <!--执行环绕增加-->
        <aop:advisor advice-ref="beforeLog" pointcut-ref="pointcut"/>
        <aop:advisor advice-ref="afterLog" pointcut-ref="pointcut"/>
    </aop:config>


</beans>

(5)测试

@Test
    public void test1() {
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
        /*
        UserServiceImpl userService = applicationContext.getBean("userService", UserServiceImpl.class)这样执行会报错
        因为动态代理代理的是接口,所以必须获取接口
         */
        UserService userService = applicationContext.getBean("userService", UserService.class);
        userService.create();
    }

测试结果如下:

 

 

 2.自定义类、自定义切面实现

接口以及实现类都与上面相同

(1)自定义切面类

package com.jms.diy;
public class diyAspect {

    public void before() {
        System.out.println("[Debug]方法执行...");
    }

    public void after() {
        System.out.println("[Debug]方法执行完成");
    }
}

切面类中自定义通知方法

(2)xml配置文件

注入切面类的Bean,配置AOP

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

    <bean id="userService" class="com.jms.service.UserServiceImpl"/>

    <!--方法二:自定义类,自定义切面-->
    <bean id="diyAspect" class="com.jms.diy.diyAspect"/>
    <aop:config>
        <!--自定义切面-->
        <aop:aspect ref="diyAspect">
            <!--切入点-->
            <aop:pointcut id="pointcut1" expression="execution(* com.jms.service.UserServiceImpl.*(..))"/>
            <aop:after method="after" pointcut-ref="pointcut1"/>
            <aop:before method="before" pointcut-ref="pointcut1"/>
        </aop:aspect>
    </aop:config>


</beans>

(3)测试如下

 

 3.通过注解实现

这种实现其实是第二种的注解方式

(1)自定义切面类

package com.jms.diy;

import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;

@Aspect
@Component
public class annotationAspect {
    @Before("execution(* com.jms.service.UserServiceImpl.*(..))")
    public void before() {
        System.out.println("[Debug]方法前置增强");
    }

    @After("execution(* com.jms.service.UserServiceImpl.*(..))")
    public void after() {
        System.out.println("[Debug]方法后置增强");
    }
}

(2)此处还是采用xml配置,也可以采用java类配置

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

    <bean id="userService" class="com.jms.service.UserServiceImpl"/>

    <!--方法三:注解-->
    <!--增加注解支持-->
    <context:annotation-config/>
    <context:component-scan base-package="com.jms"/>
    <aop:aspectj-autoproxy/>
    
</beans>

(3)测试

 

(本文仅作个人学习记录用,如有纰漏敬请指正)

有关Spring(十四):SpringAOP及AOP的三种实现方法的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  3. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  4. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  5. Ruby 方法() 方法 - 2

    我想了解Ruby方法methods()是如何工作的。我尝试使用“ruby方法”在Google上搜索,但这不是我需要的。我也看过ruby​​-doc.org,但我没有找到这种方法。你能详细解释一下它是如何工作的或者给我一个链接吗?更新我用methods()方法做了实验,得到了这样的结果:'labrat'代码classFirstdeffirst_instance_mymethodenddefself.first_class_mymethodendendclassSecond使用类#returnsavailablemethodslistforclassandancestorsputsSeco

  6. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

    我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

  7. ruby - Highline 询问方法不会使用同一行 - 2

    设置:狂欢ruby1.9.2高线(1.6.13)描述:我已经相当习惯在其他一些项目中使用highline,但已经有几个月没有使用它了。现在,在Ruby1.9.2上全新安装时,它似乎不允许在同一行回答提示。所以以前我会看到类似的东西:require"highline/import"ask"Whatisyourfavoritecolor?"并得到:Whatisyourfavoritecolor?|现在我看到类似的东西:Whatisyourfavoritecolor?|竖线(|)符号是我的终端光标。知道为什么会发生这种变化吗? 最佳答案

  8. ruby - 主要 :Object when running build from sublime 的未定义方法 `require_relative' - 2

    我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby​​1.9+ 关于ruby-主要:Objectwhenrun

  9. ruby - 多个属性的 update_column 方法 - 2

    我有一个具有一些属性的模型:attr1、attr2和attr3。我需要在不执行回调和验证的情况下更新此属性。我找到了update_column方法,但我想同时更新三个属性。我需要这样的东西:update_columns({attr1:val1,attr2:val2,attr3:val3})代替update_column(attr1,val1)update_column(attr2,val2)update_column(attr3,val3) 最佳答案 您可以使用update_columns(attr1:val1,attr2:val2

  10. ruby - 检查方法参数的类型 - 2

    我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)

随机推荐