一、采用配置方法使用AOP
(一)在net.lbj.spring包里创建lesson05.aop_xml子包
(二)创建杀龙任务类
(三)创建勇敢骑士类
(四)创建游吟诗人类
(五)创建Spring配置文件
在resources里创建aop_xml目录,在里面创建spring-config.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:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
https://www.springframework.org/schema/aop/spring-aop.xsd">
<!--组件扫描-->
<context:component-scan base-package="net.lbj.spring.lesson05.aop_xml"/>
<!--AOP配置-->
<aop:config>
<!--定义切面-->
<aop:aspect ref="minstrel">
<!--定义切点-->
<aop:pointcut id="embark" expression="execution(* net.lbj.spring.lesson05..*.embarkOnQuest(..))"/>
<!--声明前置通知-->
<aop:before method="singBeforeQuest" pointcut-ref="embark"/>
<!--声明后置通知-->
<aop:after method="singAfterQuest" pointcut-ref="embark"/>
</aop:aspect>
</aop:config>
</beans>
(六)在pom文件里添加依赖
<!--Spring AOP-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>${spring.version}</version>
</dependency>
<!--AspectJ支持-->
<dependency>
<groupId>aspectj</groupId>
<artifactId>aspectjrt</artifactId>
<version>1.5.4</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.6</version>
<scope>runtime</scope>
</dependency>
(七)创建测试类
- 测试类
package net.lbj.spring.lesson05.aop_xml;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* 测试骑士类
* 20210324
*/
public class TestKnight {
private ClassPathXmlApplicationContext context; //基于类路径XML配置文件的应用容器
@Before
public void init() {
//基于Spring配置文件创建应用容器
context = new ClassPathXmlApplicationContext("aop_xml/spring-config.xml");
}
@Test
public void testBraveKnight() {
//根据名称从应用容器中获取勇敢骑士对象
BraveKnight braveKnight = (BraveKnight) context.getBean("mike");
//勇敢骑士执行任务
braveKnight.embarkOnQuest();
}
@After
public void destroy() {
//关闭应用容器
context.close();
}
}
- 运行结果
(八)课堂练习
1、增加救美任务类与救美骑士类
- 救美骑士类
- 救美任务类
2、增加 testDamselRescuingKnight()测试方法及其运行结果
- 测试方法
- 运行结果
二、采用注解方式使用AOP
(一)在net.lbj.spring包里创建lesson05.aop_annotation子包
(二)创建杀龙任务类
(三)创建勇敢骑士类
(四)创建游吟诗人切面
package net.lbj.spring.lesson05.aop_annotation;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
/**
* 游吟诗人切面
* 20210329
*/
@Aspect
@Component //交给Spring容器管理
public class MinstrelAspect {
//注解声明切点
@Pointcut("execution(* net.lbj.spring.lesson05..*.embarkOnQuest(..))")
public void embark() {
}
//注解声明前置通知
@Before("embark()")
public void singBeforeQuest(JoinPoint joinPoint) {
System.out.println("啦啦啦,骑士出发啦!");
}
//注解声明后置通知
@After("embark()")
public void singAfterQuest(JoinPoint joinPoint) {
System.out.println("真棒啊!骑士完成了任务 !");
}
}
(五)创建Spring配置类
(六)创建测试类及其运行结果
- 测试类
package net.lbj.spring.lesson05.aop_annotation;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
/**
* 测试骑士类
* 20210329
*/
public class TestKnight {
private AnnotationConfigApplicationContext context; //基于注解配置类的应用容器
@Before
public void init() {
//基于注解配置类创建应用容器
context = new AnnotationConfigApplicationContext(AopConfig.class);
}
@Test
public void testBraveKnight() {
//根据名称从应用容器里获取勇敢骑士对象
BraveKnight braveKnight = (BraveKnight) context.getBean("mike");
//勇敢骑士执行任务
braveKnight.embarkOnQuest();
}
@Test
public void testDamselRescuingKnight() {
//根据名称从应用容器里获取救美骑士对象
DamselRescuingKnight Knight = (DamselRescuingKnight) context.getBean("damselRescuingKnight");
//救美骑士执行任务
Knight.embarkOnQuest();
}
@After
public void destroy() {
//关闭应用容器
context.close();
}
}
- testBraveKnight()运行结果
(七)课堂练习
1、增加救美任务类与救美骑士类
- 救美骑士类
- 救美任务类
2、增加 testDamselRescuingKnight()测试方法及其运行结果
- 测试类
- 运行结果
三、实现注解式拦截
(一)创建注解接口
(二)修改勇敢骑士类
给embarkOnQuest()添加自定义注解Action
(三)修改游吟诗人切面
package net.lbj.spring.lesson05.aop_annotation;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;
import java.lang.reflect.Method;
/**
* 游吟诗人切面
* 20210329
*/
@Aspect //声明为切面
@Component //交给Spring容器管理
public class MinstrelAspect {
//注解声明切点
@Pointcut("@annotation(net.lbj.spring.lesson05.aop_annotation.Action)")
public void embark() {
}
//注解声明前置通知
@Before("embark()")
public void singBeforeQuest(JoinPoint joinPoint) {
//获取方法签名
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
//获取被拦截的方法
Method method = signature.getMethod();
//获取注解式拦截
Action action = method.getAnnotation(Action.class);
//提示用户被拦截了
System.out.println("[" + action.name() + "]拦截了[" + method.getName() + "]:拦截前!");
System.out.println("啦啦啦,骑士出发啦!");
}
//注解声明后置通知
@After("embark()")
public void singAfterQuest(JoinPoint joinPoint) {
//获取方法签名
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
//获取被拦截的方法
Method method = signature.getMethod();
//获取注解式拦截
Action action = method.getAnnotation(Action.class);
//提示用户被拦截了
System.out.println("[" + action.name() + "]拦截了[" + method.getName() + "]:拦截后!");
System.out.println("真棒啊!骑士完成了任务 !");
}
}
(四)运行测试方法testBraveKnight()
(五)修改救美骑士类
给embarkOnQuest()添加自定义注解Action
(六)运行测试方法testDamselRescuingKnight()
四、课后作业
任务:输出骑士完成任务的耗时
- 创建耗时切面类
- 运行测试方法testBraveKnight()
- 运行测试方法testDamselRescuingKnight()
五、遇到的问题
1.
解决方法:
2.测试运行报错
原因:在杀龙任务类中少写了@Component注解符
3.运行测试testBraveKnight()方法报错
原因及解决方法:删除画×号的代码