网站首页 > 知识剖析 正文
吟游诗人用诗歌记载了骑士的事迹并将其进行传唱。利用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方式
- 上一篇: xml的名称空间是为什么
- 下一篇: Spring的装配方式xml
猜你喜欢
- 2024-11-23 Spring中源码中涉及的「设计模式」知多少?
- 2024-11-23 spring阅读--容器及实例化
- 2024-11-23 Java教程:学会写Starter-你就懂了SpringBoot自动配置
- 2024-11-23 SpringBoot学习笔记三之表述层
- 2024-11-23 如何使用Java API操作HDFS系统?
- 2024-11-23 Spring IOC容器XML配置示例
- 2024-11-23 xml 配置文件规范 校验
- 2024-11-23 阿里P7大神,Java学习之路-IO流与XML,超赞分享
- 2024-11-23 Java基础教程:dubbo源码解析-服务暴露与发现
- 2024-11-23 解决springboot的pom.xml文件第一行报错问题
- 最近发表
- 标签列表
-
- xml (46)
- css animation (57)
- array_slice (60)
- htmlspecialchars (54)
- position: absolute (54)
- datediff函数 (47)
- array_pop (49)
- jsmap (52)
- toggleclass (43)
- console.time (63)
- .sql (41)
- ahref (40)
- js json.parse (59)
- html复选框 (60)
- css 透明 (44)
- css 颜色 (47)
- php replace (41)
- css nth-child (48)
- min-height (40)
- xml schema (44)
- css 最后一个元素 (46)
- location.origin (44)
- table border (49)
- html tr (40)
- video controls (49)