文章目录
- 一、采用配置方式使用AOP
- 二、课堂练习
- 二、采用注解方式使用AOP
- 1、在net.hw.spring包里创建lesson05.aop_annotation子包
- 2、在aop_annotation子包里创建杀龙任务类 - SlayDragonQuest
- 3、在aop_annotation子包里创建勇敢骑士类 - BraveKnight
- 4、在aop_annotation子包里创建游吟诗人切面 - MinstrelAspect
- 5、在aop_annotation子包里创建Spring配置类 - AopConfig
- 6、在test/java/net.yc.spring.lesson05里面创建aop_annotation子包,在aop_annotation子包里创建测试类 - TestKnight
- 7、运行测试
- 8、课堂练习
- 9、实现注解式拦截
- 9、课后作业
一、采用配置方式使用AOP
1、在net.hw.spring包里创建lesson05.aop_xml子包
2、在aop_xml子包里创建杀龙任务类 - SlayDragonQuest
3、在aop_xml子包里创建勇敢骑士类 - BraveKnight
4、在aop_xml子包里创建游吟诗人类 - Minstrel
5、创建Spring配置文件
在resources里创建aop_xml目录,在里面创建spring-config.xml配置文件
解析图
(1)切点
在使用Spring框架配置AOP时,不管是通过XML配置文件还是注解方式,都需要定义pointcut(切点)。
(2)切点表达式
定义切点表达式execution (* net.hw.spring….(…))
(3)切换函数
execution()是最常用的切点函数,整个表达式可以分为五个部分。
execution():表达式主体。
第一个*号:表示返回类型,号表示所有的类型。
包名:表示需要拦截的包名,后面的两个句点表示当前包和当前包的所有子包,net.hw.spring包、子孙包下所有类的方法。
第二个号:表示类名,*号表示所有的类。
*(…):最后这个星号表示方法名,*号表示所有的方法,后面括弧里面表示方法的参数,两个句点表示任何参数。
6、在pom.xml文件里添加AOP相关依赖
<!--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>
7、创建测试类- TestKnight
在test/java里创建net.yc.spring.lesson05.aop_xml包,在包里创建TestKnight
8、运行测试
二、课堂练习
1、创建拯救少女类和拯救少女骑士类
2、在TestKnight里面创建testDamselRescuingKnight()方法
3、运行testDamselRescuingKnight()方法
二、采用注解方式使用AOP
1、在net.hw.spring包里创建lesson05.aop_annotation子包
2、在aop_annotation子包里创建杀龙任务类 - SlayDragonQuest
3、在aop_annotation子包里创建勇敢骑士类 - BraveKnight
4、在aop_annotation子包里创建游吟诗人切面 - MinstrelAspect
和配置文件对比
package net.yc.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;
/**
*游吟诗人切面类
* */
@Aspect//切面
@Component//spring容器管理
public class MinstrelAspect {
//注解声明切点
@Pointcut("execution(* net.yc.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("真棒啊!骑士完成了任务!");
}
}
5、在aop_annotation子包里创建Spring配置类 - AopConfig
6、在test/java/net.yc.spring.lesson05里面创建aop_annotation子包,在aop_annotation子包里创建测试类 - TestKnight
7、运行测试
错误: 警告: Exception encountered during context initialization - cancelling refresh attempt: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name ‘Mike’: Unsatisfied dependency expressed through field ‘slayDragonQuest’; nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type ‘net.yc.spring.lesson05.aop_xml.SlayDragonQuest’ available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}
原因:在使用cv大法拷贝上一讲的类时导错包
改正后:
8、课堂练习
1、增加拯救少女任务类与拯救少女骑士类
注意:嫌麻烦要直接拷贝上一讲的类时,不要导错包
2、在测试程序里增加对拯救少女骑士的测试方法 - testDamselRescuingKnight()
运行
9、实现注解式拦截
1、在aop_annotation子包里创建注解接口 - Action
(1)@Target({ElementType.TYPE}) 注解
ElementType 这个枚举类型的常量提供了一个简单的分类:注解可能出现在Java程序中的语法位置(这些常量与元注解类型(@Target)一起指定在何处写入注解的合法位置)
(2)@Retention({RetentionPolicy.Runtime}) 注解
RetentionPolicy这个枚举类型的常量描述保留注解的各种策略,它们与元注解(@Retention)一起指定注释要保留多长时间
(3)@Documented注解
Documented注解表明这个注解是由 javadoc记录的,在默认情况下也有类似的记录工具。 如果一个类型声明被注解了文档化,它的注解成为公共API的一部分。
2、修改勇敢骑士类,给embarkOnQuest()添加自定义注解Action
3、修改游吟诗人切面类 - MinstrelAspect
package net.yc.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;
/**
*游吟诗人切面类
* */
@Aspect//切面
@Component//spring容器管理
public class MinstrelAspect {
//注解声明切点
// @Pointcut("execution(* net.yc.spring.lesson05..*.embarkOnQuest(..))")
// public void embark() {
// }
// //声明前置通知
// @Before("embark()")
// public void singBeforeQuest(JoinPoint joinPoint) {
// System.out.println("啦啦啦,骑士出发了!");
// }
// 注解声明切点(拦截添加了Action注解的方法)
@Pointcut("@annotation(net.yc.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("真棒啊!骑士完成了任务!");
}
}
4、运行测试方法testBraveKnight(),查看效果
5、修改拯救少女骑士类,给embarkOnQuest()添加自定义注解Action
6、运行测试方法testDamselRescuingKnight(),查看效果
9、课后作业
任务:输出骑士完成任务的耗时
1、创建耗时切面类 - ElapseAspect
package net.yc.spring.lesson05.aop_annotation;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.*;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;
import java.lang.reflect.Method;
/**
* 任务耗时切面类
* */
@Aspect//切面
@Component//spring容器管理
public class ElapseAspect {
long start;
long end;
@Pointcut("@annotation(net.yc.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);
start = System.currentTimeMillis();
}
@After("embark()")
public void singAfterQuest(JoinPoint joinPoint){
//获取方法签名
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
//获取被拦截的方法
Method method = signature.getMethod();
//获取注解式拦截
Action action = method.getAnnotation(Action.class);
end = System.currentTimeMillis();
System.out.println("任务耗时:"+(end-start)+"ms");
}
}