百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

Spring的AOP示例之xml方式

xsobi 2024-11-24 23:35 1 浏览

吟游诗人用诗歌记载了骑士的事迹并将其进行传唱。利用AOP,你可以声明吟游诗人必须歌颂骑士的探险事迹,而骑士本身并不用直接访问吟游诗人的方法。

切面

要将吟游诗人抽象为一个切面,你只需在一个Spring配置文件中声明它:

src/main/resources/spring/吟游诗人.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"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/aop
      http://www.springframework.org/schema/aop/spring-aop.xsd
		http://www.springframework.org/schema/beans
      http://www.springframework.org/schema/beans/spring-beans.xsd">

    <!--程序清单1.11 将 系统服务.吟游诗人 声明为一个切面-->
    <bean id="骑士" class="勇敢的骑士">
        <constructor-arg ref="任务" />
    </bean>

    <bean id="任务" class="屠龙任务">
        <constructor-arg value="#{T(System).out}" />
    </bean>

    <!-- 1. 首先,需要把`系统服务.吟游诗人`声明为一个 Bean-->
    <bean id="吟游诗人" class="系统服务.吟游诗人">
        <constructor-arg value="#{T(System).out}" />
    </bean>

    <aop:config>
        <!--  2. 然后在 `<aop:aspect>` 元素中引用该 Bean。-->
        <aop:aspect ref="吟游诗人">
            <!--  3. 切点-->
            <aop:pointcut id="embark"
                          expression="execution(* *.执行任务(..))"/>
            <!-- 4. 前置通知(before advice)-->
            <aop:before pointcut-ref="embark"
                        method="调用于任务之前"/>
            <!-- 5. 后置通知(before advice)-->
            <aop:after pointcut-ref="embark"
                       method="调用于任务之后"/>
        </aop:aspect>
    </aop:config>

</beans>

这里使用了Spring的aop配置命名空间把吟游诗人 Bean声明为一个切面。

  • 首先,需要把吟游诗人声明为一个 Bean, <!-- 1. 首先,需要把`系统服务.吟游诗人`声明为一个 Bean--> <bean id="吟游诗人" class="系统服务.吟游诗人"> <constructor-arg value="#{T(System).out}" /> </bean>
  • 然后在 <aop:aspect> 元素中引用该 Bean。 <aop:config> <!-- 2. 然后在 `<aop:aspect>` 元素中引用该 Bean。--> <aop:aspect ref="吟游诗人"> <!-- 3. 切点--> <aop:pointcut id="embark" expression="execution(* *.执行任务(..))"/> <!-- ...-->
  • 为了进一步定义切面,声明(使用<aop:before>)在执行任务()执行前调用吟游诗人调用于任务之前()。这种方式被称为前置通知(before advice)。 <!-- 4. 前置通知(before advice)--> <aop:before pointcut-ref="embark" method="调用于任务之前"/>
  • 同时声明(使用 <aop:after> )在着手任务()执行后调用调用于任务之后()。这种方式被称为后置通知 (after advice)。 <!-- 5. 后置通知(before advice)--> <aop:after pointcut-ref="embark" method="调用于任务之后"/>
  • 在这两种方式中,pointcut-ref属性都引用了名字为 embark 的切入点。该切入点是在前边的 <aop:pointcut> 元素中定义的,并配置expression属性来选择所应用的通知。表达式的语法采用的是AspectJ的切点表达式语言。 <!-- 3. 切点--> <aop:pointcut id="embark" expression="execution(* *.执行任务(..))"/>

现在你已经知道,Spring在骑士执行探险任务前后会调用 吟游诗人调用于任务之前()调用于任务之后()

这就是我们需要做的所有的事情!少量的XML配置,就可以把 吟游诗人 声明为一个Spring切面。

我们可以从这个示例中获得两个重要的观点。

  • 首先,吟游诗人 仍然是一个POJO,没有任何代码表明它要被作为一个切面使用。当我们按照上面那样进行配置后,在Spring的上下文中,吟游诗人 实际上已经变成一个切面了。
  • 其次,也是最重要的,吟游诗人 可以被应用到 勇敢的骑士 中,而 勇敢的骑士 不需要显式地调用它。实际上, 勇敢的骑士 完全不知道吟游诗人 的存在。

还要指出的是,尽管我们使用Spring魔法把吟游诗人转变为一个切面,但首先要把它声明为一个Spring Bean。能够为其他Spring Bean做到的事情都可以同样应用到Spring切面中,例如为它们注入依赖。

dependencies

    <dependencies>
        <dependency>
            <!--  AspectJ的切点表达式语言 支持 -->
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.7.2</version>
            <scope>compile</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-test</artifactId>
            <version>4.0.7.RELEASE</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>4.0.7.RELEASE</version>
            <scope>compile</scope>
        </dependency>
    </dependencies>

运行

import org.springframework.context.support.ClassPathXmlApplicationContext;

// 加载包含 骑士 的Spring上下文
public class 骑士Main {
  public static void main(String[] args) throws Exception {
    ClassPathXmlApplicationContext context = 
        new ClassPathXmlApplicationContext(
                "spring/吟游诗人.xml");
    I骑士 某骑士 = context.getBean(I骑士.class);
    某骑士.执行任务();
    context.close();
  }
}

结果:

Fa la la, the 骑士真勇敢!
踏上屠龙的征程!
Tee hee hee, 勇敢的骑士执行完任务!

这就是我们需要做的所有的事情!少量的XML配置,就可以把 吟游诗人 声明为一个Spring切面。

其他代码

吟游诗人

package 系统服务;

import java.io.PrintStream;

// 吟游诗人是中世纪的音乐记录器
public class 吟游诗人 {

  private PrintStream stream;
  
  public 吟游诗人(PrintStream stream) {
    this.stream = stream;
  }

  public void 调用于任务之前() {
    stream.println("Fa la la, the 骑士真勇敢!");
  }

  public void 调用于任务之后() {
    stream.println("Tee hee hee, 勇敢的骑士执行完任务!");
  }
}

吟游诗人是只有两个方法的简单类。在骑士执行每一个探险任务之前,调用于探险之前()被调用;在骑士完成探险任务之后,调用于探险之后()被调用。在这两种情况下,吟游诗人都会通过一个PrintStream类来歌颂骑士的事迹,这个类是通过构造器注入的。

任务

public interface I任务 {
  void 执行();
}
import java.io.PrintStream;

public class 屠龙任务 implements I任务 {

    private PrintStream stream;

    public 屠龙任务(PrintStream stream) {
        this.stream = stream;
    }

    public void 执行() {
        stream.println("踏上屠龙的征程!");
    }
}

骑士

public interface I骑士 {
  void 执行任务();
}
public class 勇敢的骑士 implements I骑士 {

  private I任务 任务;

  public 勇敢的骑士(I任务 任务) {
    this.任务 = 任务;
  }

  public void 执行任务() {
    任务.执行();
  }
}

全部的代码可以在以下位置找到

https://gitee.com/virhuiai/spring-wiring-example/tree/master/AOP示例之xml方式

相关推荐

js向对象中添加元素(对象,数组) js对象里面添加元素

一、添加一个元素对象名["属性名"]=值(值:可以是一个值,可以是一个对象,也可以是一个数组)这样添加进去的元素,就是一个值或对象或数组...

JS小技巧,如何去重对象数组?(一)

大家好,关于数组对象去重的业务场景,想必大家都遇到过类似的需求吧,这对这样的需求你是怎么做的呢。下面我就先和大家分享下如果是基于对象的1个属性是怎么去重实现的。方法一:使用.filter()和....

「C/C++」之数组、vector对象和array对象的比较

数组学习过C语言的,对数组应该都不会陌生,于是这里就不再对数组进行展开介绍。模板类vector模板类vector类似于string,也是一种动态数组。能够在运行阶段设置vector对象的长度,可以在末...

如何用sessionStorage保存对象和数组

背景:在工作中,我将[{},{}]对象数组形式,存储到sessionStorage,然后ta变成了我看不懂的形式,然后我想取之用之,发现不可能了~记录这次深刻的教训。$clickCouponIndex...

JavaScript Array 对象 javascript的array对象

Array对象Array对象用于在变量中存储多个值:varcars=["Saab","Volvo","BMW"];第一个数组元素的索引值为0,第二个索引值为1,以此类推。更多有...

JavaScript中的数组Array(对象) js array数组

1:数组Array:-数组也是一个对象-数组也是用来存储数据的-和object不同,数组中可以存储一组有序的数据,-数组中存储的数据我们称其为元素(element)-数组中的每一个元素都有一...

数组和对象方法&amp;数组去重 数组去重的5种方法前端

列举一下JavaScript数组和对象有哪些原生方法?数组:arr.concat(arr1,arr2,arrn);--合并两个或多个数组。此方法不会修改原有数组,而是返回一个新数组...

C++ 类如何定义对象数组?初始化数组?linux C++第43讲

对象数组学过C语言的读者对数组的概念应该很熟悉了。数组的元素可以是int类型的变量,例如int...

ElasticSearch第六篇:复合数据类型-数组,对象

在ElasticSearch中,使用JSON结构来存储数据,一个Key/Value对是JSON的一个字段,而Value可以是基础数据类型,也可以是数组,文档(也叫对象),或文档数组,因此,每个JSON...

第58条:区分数组对象和类数组对象

示例设想有两个不同类的API。第一个是位向量:有序的位集合varbits=newBitVector;bits.enable(4);bits.enable([1,3,8,17]);b...

八皇后问题解法(Common Lisp实现)

如何才能在一张国际象棋的棋盘上摆上八个皇后而不致使她们互相威胁呢?这个著名的问题可以方便地通过一种树搜索方法来解决。首先,我们需要写一个函数来判断棋盘上的两个皇后是否互相威协。在国际象棋中,皇后可以沿...

visual lisp修改颜色的模板函数 怎么更改visual studio的配色

(defunBF-yansemokuai(tuyuanyanse/ss)...

用中望CAD加载LISP程序技巧 中望cad2015怎么加载燕秀

1、首先请加载lisp程序,加载方法如下:在菜单栏选择工具——加载应用程序——添加,选择lisp程序然后加载,然后选择添加到启动组。2、然后是添加自定义栏以及图标,方法如下(以...

图的深度优先搜索和广度优先搜索(Common Lisp实现)

为了便于描述,本文中的图指的是下图所示的无向图。搜索指:搜索从S到F的一条路径。若存在,则以表的形式返回路径;若不存在,则返回nil。...

两个有助于理解Common Lisp宏的例子

在Lisp中,函数和数据具有相同的形式。这是Lisp语言的一个重大特色。一个Lisp函数可以分析另一个Lisp函数;甚至可以和另一个Lisp函数组成一个整体,并加以利用。Lisp的宏,是实现上述特色的...