假如我们的系统的很多模块都需要实现一些相同的功能,这些相同的功能被称为横切关注点,在切面出现之前,我们可以使用继承或委托来实现这些相同的功能,但是这种做法会增加系统不同模块之间的耦合,关于AOP的术语,有下面一些
1、通知,定义了切面是什么以及何时使用,通知的类型包括前置通知、后置通知、返回通知、异常通知和环绕通知
2、连接点,是指应用在执行过程中能够插入的一个点
3、切点,定义了切面的工作范围
4、切面,是通知和切点的结合
5、引入,引入允许我们向现有的类添加新的方法或属性
6、织入,把切面应用到目标对象并创建新的代理对象的过程时,在目标对象的多个生命周期内,有多个点可以织入,包括编译期、类加载期和运行期,Spring是在运行期采取织入这种方法的
Spring对AOP的支持
Spring提供了4中类型的AOP支持
1、基于代理的经典Spring AOP
2、纯POJO切面
3、@AspectJ注解驱动的切面
4、注入式AspectJ切面
在Spring中使用AspectJ进行制造切点时,我们使用AspectJ的切点表达式语言来定义Spring切面,Spring AOP只支持下面的几种AspectJ指示器
AspectJ指示器 | 描述 |
---|---|
arg() | 限制连接点匹配参数为指定类型的执行方法 |
@args() | 限制连接点匹配参数由指定注解标注的执行方法 |
execution() | 用于匹配是连接点的执行方法 |
this() | 限制连接点匹配AOP代理的bean引用为指定类型的类 |
target | 限制连接点匹配目标对象为指定类型的类 |
@target() | 限制连接点匹配特定的对象,这些对象对应的类要具有指定类型的注解 |
within() | 限制连接点匹配的指定的类型 |
@within() | 限制连接点匹配指定注解所标注的类型 |
@annotation | 限定匹配带有指定注解的连接点 |
切面的编写格式如下
execution(* com.fan.aop.Performance.perform(..))
具体的解释如下
1、*,返回值部分,在这里表示任意的返回值
2、com.fan.aop.Performance.perform,表示织入切面的方法
3、(..),参数部分,在这里参数为任意参数
4、execution,表示触发的时机,execution表示在执行的时候触发
切面的实例
首先我们创建一个接口
public interface Performance {
public void perform();
}
实现该接口为Dancer
@Component
public class Dancer implements Performance {
@Override
public void perform() {
System.out.println("Dancing Liking a Swan");
}
}
创建一个切面的POJO
@Aspect
public class Audience {
@Before("execution(* com.fan.aop.Performance.perform(..))")
public void silenceCellPhone(){
System.out.println("Silence cell phone");
}
@Before("execution(* com.fan.aop.Performance.perform(..))")
public void takeSeats(){
System.out.println("Take seats");
}
@AfterReturning("execution(* com.fan.aop.Performance.perform(..))")
public void applause(){
System.out.println("Applause!");
}
@AfterThrowing("execution(* com.fan.aop.Performance.perform(..))")
public void demandRefund(){
System.out.println("Demand refund");
}
}
使用JavaConfig来进行配置
@Configuration
@EnableAspectJAutoProxy
@ComponentScan
public class ConcertConfig {
@Bean
public Audience audience(){
return new Audience();
}
}
测试并查看结果
public class Main {
public static void main(String[] args){
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ConcertConfig.class);
// ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("AOPConfig.xml");
Performance dancer = context.getBean(Performance.class);
dancer.perform();
context.close();
}
}
看上面,我们发现Audience 的实现中有一些是重复的,因此我们可以使用@Ponitcut注解来进行管理,重写如下
@Aspect
public class AudienceBeta {
@Pointcut("execution(* com.fan.aop.Performance.perform(..))")
public void performance(){}
@Before("performance()")
public void silenceCellPhone(){
System.out.println("Silence cell phone");
}
@Before("performance()")
public void takeSeats(){
System.out.println("Take seats");
}
@AfterReturning("performance()")
public void applause(){
System.out.println("Applause!");
}
@AfterThrowing("performance()")
public void demandRefund(){
System.out.println("Demand refund");
}
}
除了使用JavaConfig的方式来配置切面,我们也可以使用XML文件的方法来进行配置,如下
测试的地方只需要修改获取bean的方式就可以了,在这里是将
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ConcertConfig.class);
替换为
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("AOPConfig.xml");
在切面方法中,我们注意有使用多个AspectJ的注解,它们的意义如下
注解 | 通知 |
---|---|
@After | 通知方法会在目标方法返回或抛出异常后调用 |
@AfterReturning | 通知方法会在目标方法返回后调用 |
@AfterThrowing | 通知方法会在目标方法抛出异常后调用 |
@Around | 通知方法将目标方法封装起来 |
@Before | 通知方法会在目标方法调用之前执行 |
通知中包含参数的写法
@Pointcut("execution(* com.fan.soundsystem.CompactDisc.playTrack(int)) && args(trackNumber)")
public void trackPlayed(int trackNumber){}
使用XML进行配置
创建无注解的类
public class AudienceGama {
public void silenceCellPhone(){
System.out.println("Silencing cell phone");
}
public void takeSeats(){
System.out.println("Taking seats");
}
public void applause(){
System.out.println("CLAP CLAP CLAP!!!");
}
public void demandRefund(){
System.out.println("Demanding a refund");
}
}
创建对应的XML文件
创建测试代码
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("XmlAopConfig.xml");
Performance performance = context.getBean(Performance.class);
performance.perform();
context.close();
运行并查看结果