第四章 面向切面的Spring

2018-08-20  本文已影响0人  施瓦

第四章 面向切面的Spring

[TOC]

面向切面要解决的问题

在软件开发中,散布于应用中多处的功能被称为 横切关注点,例如事务、安全、日志、权限控制………通常来讲,这些横切关注点从概念上是与业务的应用逻辑相分离的。把这些横切关注点与业务逻辑相分离正是面向切面编程所要解决的问题。

横切关注点可以被模块化为特殊的类,这些类被称为切面

面向切面常用术语

描述切面的常用术语有 :通知、切点和连接点 。

通知、切点和连接点

通知

在AOP术语中,切面(横切关注点可以被模块化为特殊的类)的工作被称为通知。通知定义了 切面是什么以及何时使用,除了描述切面所完成的工作,通知还解决了何时执行这个工作的问题

Spring切面可以应用五种类型的通知 :

连接点

连接点就是在 应用执行过程中 能够 插入切面 的一个点

切点

一个切面不需要通知应用的所有连接点。切点有助于缩小切面所通知的连接点。如果说通知定义了切点是什么以及切点在何时使用的话,那么切点就定义了“何处”。切点的定义会匹配通知所要 织入 的一个或者多个连接点。

切面

切面是通知和切点的结合。通知和切点共同定义了切面的全部内容——它是什么,在何时和何处完成其功能。

引入

引入允许我们向现有类添加新方法或者属性。在无需修改现有的类的情况下,让他们具有新的行为和状态。

织入

织入是把切面应用到目标对象并创建新的代理的过程。切面在指定的连接点被织入到目标对象中。在目标对象的生命周期里有多个点可以进行织入 :

总结

通知包含了需要用于多个应用对象的横切行为(横切关注点);连接点是程序执行过程中能够应用通知的所有点;切点定义了通知被应用的具体位置(在哪些连接点会得到通知)。

Spring对AOP的支持

Spring提供了四种类型的AOP支持 :

Spring通知是使用Java编写的,定义通知所应用的切点通常会使用注解或XML编写。

通过在代理类中包裹切面(通知 + 切点),Spring在运行期把切面织入到Spring管理的bean中。如图,代理类封装了目标类,并拦截被通知方法的调用,再把调用转发给真正的目标bean。当代理拦截到方法调用时,在调用目标bean方法之前,会执行切面逻辑

SPring的切面由包裹了目标对象的代理类实现。代理类处理方法的调用,执行额外的切面逻辑,并调用目标方法

直到应用需要被代理的bean时,Spring才会创建代理对象。

因为Spring基于动态代理,所以Spring只支持方法连接点。Spring缺少对字段连接点的支持,无法让我们创建细粒度的通知,例如拦截对象字段的修改。

通过切点来选择连接点

使用AspectJ的切点表达式语言

在Spring AOP中,使用AspectJ的切点表达式语言定义切点,其中execution是最重要的描述符 :

execution(modifiers-pattern? ret-type-pattern declaring-type-pattern?name-pattern(param-pattern)
            throws-pattern?)

除了返回类型、方法名称以及参数列表之外,其余都是可选的(即含有?的都是可选的)

例如 :

匹配所有的public方法 :

execution(public * *(..))

匹配所有方法名开头为set的 :

execution(* set*(...))

匹配定义在AccountService接口类中的所有方法

execution(* com.xyz.service.AccountService.*(...))

匹配定义在service包下的所有方法

execution(* com.xyz.service.*.*(..))

匹配定义在service包或者子包下的所有方法

execution(* com.xyz.service..*.*(..))
Aspect指示器 描述
execution() 用于匹配连接点
arg() 表明连接点参数类型是匹配类
@args() 表明参数注解是匹配类
this() 匹配一个bean,这个bean是一个指定类型的实例
target 匹配一个目标对象,此对象是一个给定类型的实例
@target() 匹配对象类需要有指定类型的注解
within() 限制连接点匹配指定的类型
@within() 匹配方法,该方法需要给定一个特定注解
@annotation 匹配带有指定注解的连接点

编写切点

假设我们需要编写Performance类型 :

public interface Performance{
    public void perform();
}

我们想要编写一个切面,在调用Performance类中的perform方法时触发通知 :

execution(public * concert.Performance.perform(..))
image

现在我们假设我们需要配置的切点仅仅匹配concert包 :

execution(public * concert.Performance.perform(..) && within(concert.*))

当使用Spring的XML来描述切面时候,我们可以使用and来替换&&,同样的,ornot可以替换||!

在切点中选择bean

Spring中的bean()指示器允许我们在切点表达式中使用bean的ID来标识bean。bean()使用bean IDbean名称作为参数来限制切点只匹配特定的bean

例如 :

execution(* concert.Performance.perform(..) and bean('woodstock'))

在上面的例子中,我们希望在执行perform()方法时应用通知,但是限制bean的ID为woodstock

使用注解创建切面

定义切面

@Aspect
public class Audience {
    // 演出之前
    @Before("execution(public * concert.Performance.perform(..))")
    public void silenceCellPhone(){
        System.out.println("手机静音");
    }
    // 演出之前
    @Before("execution(public * concert.Performance.perform(..))")
    public void takeSeats(){
        System.out.println("对号入座");
    }
    // 演出成功之后
    @AfterReturning("execution(public * concert.Performance.perform(..)")
    public void applause(){
        System.out.println("掌声雷动");
    }
    // 演出失败之后
    @AfterThrowing("execution(public * concert.Performance.perform(..))")
    public void demandRefund(){
        System.out.println("我想退款!");
    }

}

Audience类使用了@Aspect注解进行标注,表明该类不仅是一个POJO,还是一个切面Audience类中的方法都是用注解来定义切面的具体行为。

AspectJ使用了五个注解来定义通知 :

注解 通知
@After 通知方法在目标方法返回或者抛出异常时调用
@AfterReturning 通知方法在目标方法返回后调用
@AfterThrowing 通知方法在目标方法抛出异常后调用
@Around 通知方法会将目标方法封装起来
@Before 通知方法在目标方法调用之前调用

在上面的例子中,我们定义了四个切点表达式,这四个表达式完全可以进行整合 :

PointCut注解能够在一个@Aspect切面内定义可以重复的切点

@Aspect
public class Audience {
    @Pointcut("execution(public * concert.Performance.perform(..))")
    public void perform(){}
    // 演出之前
    @Before("perform()")
    public void silenceCellPhone(){
        System.out.println("手机静音");
    }
    // 演出之前
    @Before("perform()")
    public void takeSeats(){
        System.out.println("对号入座");
    }
    // 演出成功之后
    @AfterReturning("perform()")
    public void applause(){
        System.out.println("掌声雷动");
    }
    // 演出失败之后
    @AfterThrowing("perform()")
    public void demandRefund(){
        System.out.println("我想退款!");
    }
}

我们还需要启动AspectJ的自动代理 :

如果你使用JavaConfig注解的话,你可以在配置类上加上@EnableAspectJAutoProxy注解启动自动代理的功能

// 启动AspectJ自动代理
@EnableAspectJAutoProxy
@Configuration
public class ConcertConfig {
    @Bean
    public Audience audience(){
        return new Audience();
    }
}

如果使用XML装配的话,我们需要<aop:aspectj-autoproxy />启动自动代理 :

<bean id="audience" class="concert.Audience"/>
<bean id="musicPerformance" class="concert.MusicPerformance"/>
<!-- 开启自动代理 -->
<aop:aspectj-autoproxy/>

Spring的AspectJ自动代理仅仅使用@Aspect作为创建切面的指导,切面依然是基于代理的。在本质上,它依然是Spring基于代理的切面。这一点非常重要,因为这意味着尽管使用的是@Aspect注解,但是仍然限于代理方法的调用。如果想使用AspectJ的所有能力,我们必须在运行时使用AspectJ并且不依赖Spring来创建基于代理的切面。

创建环绕通知

环绕通知能够让你所编写的逻辑将被通知的目标方法(连接点)完全包裹起来。就像是在一个通知方法中同时编写前置后置通知。

@Aspect
public class Audience {

    @Pointcut("execution(public * concert.Performance.perform(..))")
    public void perform(){}

    // 环绕通知
    @Around("perform()")
    public void watchPerformance(ProceedingJoinPoint joinPoint){
        try{
            System.out.println("关闭手机");
            System.out.println("入座");
            // 通过ProceedingJoinPoint来调用被通知的方法
            joinPoint.proceed();
            System.out.println("掌声雷动");
        }catch(Throwable e){
            System.out.println("我要退款");
        }
    }
}

@Around注解表明watchPerformance()方法会作为performance切点的环绕通知。当通知方法需要把控制权交给被通知方法时候,需要调用ProceedingJoinPointproceed()方法。如果不调用这个方法的话,你的通知会阻塞对被通知方法的调用。

为通知传递参数

BlankDisc中,我们需要统计磁道被播放的数量 :


public class BlankDisc implements CompactDisc {

    private String title;
    private String artist;
    private List<String> tracks;
    public BlankDisc(String title,String artist,List<String> tracks) {
        this.title = title;
        this.artist = artist;
        this.tracks = tracks;
    }
    public String getTitle() {
        return title;
    }
    public String getArtist() {
        return artist;
    }
    public List<String> getTracks() {
        return tracks;
    }
    public void play() {
        System.out.println("title :" + title + " artist :" + artist);
        for (int trackNumber = 0;trackNumber < tracks.size();trackNumber ++){
            playTrack(trackNumber);
        }
    }
    public void playTrack(int trackNumber) {
        System.out.println("track "+ trackNumber + " : " + tracks.get(trackNumber));
    }
}

我们定义TrackCounter来描述切面 :

@Aspect
@Component
@EnableAspectJAutoProxy
public class TrackCounter {
    public Map<Integer,Integer> trackCounts = new HashMap<Integer, Integer>();
    @Pointcut("execution(public * soundsystem.BlankDisc.playTrack(int)) && args(trackNumber))")
    public void trackPlayed(int trackNumber){}
    @AfterReturning("trackPlayed(trackNumber)")
    public void countTrack(int trackNumber){
        trackCounts.put(trackNumber,getPlayCount(trackNumber) + 1);
        System.out.println("--->track " + trackNumber + "数量增加了.");
    }
    public int getPlayCount(int trackNumber){
        return trackCounts.containsKey(trackNumber)?
                trackCounts.get(trackNumber):0;
    }
}

切点表达式中的args(trackNumber)表明 :传递给连接点的int类型的参数也会传递到通知方法中。参数的名称为trackNumber,与切点方法签名中的参数相匹配。在@AfterReturing("trackNumber")表达式下面,切点方法和切点定义的参数名一致。

通过注解引入新功能

TODO

在XML中声明切面

前置后置通知

<!-- 切面配置 -->
<!-- 顶层的aop配置元素 -->    
<aop:config>
        <!-- 定制一个切面 -->
        <aop:aspect ref="audience">
            <!-- 定义一个切点 -->
            <aop:pointcut id="perform" expression="execution(public * concert.Performance.perform(..))"/>

            <aop:before method="takeSeats" pointcut-ref="perform"/>

            <aop:before method="silenceCellPhone" pointcut-ref="perform"/>

            <aop:after-returning method="applause" pointcut-ref="perform"/>

            <aop:after-throwing method="demandRefund" pointcut-ref="perform"/>

        </aop:aspect>
    </aop:config>

环绕通知

<bean id="musicPerformance" class="concert.MusicPerformance"/>
<bean id="audience" class="concert.Audience"/>
<aop:config>
    <aop:aspect ref="audience">
    <aop:pointcut id="performance" expression="execution(public * concert.Performance.perform())"/>
    <aop:around method="execute" pointcut-ref="performance"/>
        </aop:aspect>
</aop:config>

为通知传递参数

<beans> 
    <bean id="blankDisc" class="soundsystem.BlankDisc"
          c:_0="${disc.title}" c:_1="${disc.artist}" c:_2-ref="blankDiscList"/>

    <bean id="cdPlayer" class="soundsystem.CDPlayer"/>

    <bean id="trackCounter" class="soundsystem.TrackCounter"/>

    <util:list id="blankDiscList">
        <value>老古董</value>
        <value>大千世界</value>
        <value>如约而至</value>
        <value>柳成荫</value>
    </util:list>

    <context:property-placeholder location="classpath:/application.properties"/>

    <aop:aspectj-autoproxy/>

    <import resource="classpath:/aopconfig.xml"/>
</beans>
<beans>
    <aop:config>
        <aop:aspect ref="trackCounter">
        <aop:pointcut id="playTrack" expression="execution(* soundsystem.BlankDisc.playTrack(int)) and args(trackNumber)"/>
        <aop:after-returning pointcut-ref="playTrack" method="countTrack"/>
        </aop:aspect>
    </aop:config>
</beans>

通过切面引入新的功能

TODO

注入AspectJ切面

TODO

上一篇下一篇

猜你喜欢

热点阅读