ssm框架Java学习笔记SpringFramework

Spring AOP 笔记三(引入Introductions)

2016-09-03  本文已影响1271人  希尔大

此文是Sping in Action 第4版 英文原版切面部分的读书笔记,仅限交流使用,有不足之处,一定听取修改。

系列目录:
Spring AOP 笔记一(基础概念,一个简单切面)
Spring AOP 笔记二(环绕通知,切面中获取参数)
Spring AOP 笔记三(切面注解引入新的方法)

1. 注解引入(Annotating introductions)

在一些高级语言中,比如Rudy与Groovy,都有开放类的概念,能够在不改变对象和类的代码的前提下,为对象和类添加新的方法。然而对于Java,很不幸,他不是动态的,类一旦编译完成,就很难再为他添加新的功能。
但是你可以稍微想一想我们再次之前写的切面。虽然没有为向对象中添加新的方法,但是已经向其中添加了新的功能。既然我们能够向现有的方法添加新的功能,为什么不能为一个对象添加新的方法那?实际上,使用名为 引入(introduction) 的AOP概念,就能够实现。

之前好像没有提到切面实现的原理。切面是用到了叫做 代理 的设计模式。这里简单比喻一下,代理就好像是艺人(被切入的bean的方法)尽心尽责的经纪人[笑脸],在日常工作中,外部的通告或者节目邀请都会先送到经纪人的手里,经纪人会自己帮助艺人处理完一些必须事务(@before),,不需要艺人关系或者插手,然后经纪人处理完后,再告知(调用)艺人,去演戏或者参加节目,例如下图:

基本的代理

感兴趣的同学可以深入学习一下。

于是乎当我们想在对象中加入新的方法时,比如当前的艺人擅长表演,而有些节目想让他献声,此时经纪人就可以找个会唱歌的艺人(一个新的接口,里面有我们想要新增得唱歌方法),当栏目索要(调用)歌曲时,此时经纪人就可以直接让唱歌的艺人唱一首歌送个栏目,在栏目(调用者)那里,并不会知道是谁完成可唱歌这个任务,如下图:

添加新的方法

这样说应该差不多能理解。

然后看一个小栗子:

我们先定义一个Singer接口,代表会唱歌的艺人。
/* 
* 一个歌者 
* Created by Henvealf on 2016/9/3. 
*/
public interface Singer { 
        void sing();
}
然后新定义一个切面,类名为SingerIntroducer,歌手引入者,也就是经纪人。
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.DeclareParents;
/** 
* 歌手引入者 
* Created by Henvealf on 2016/9/3. 
*/
@Aspectpublic 
class SingerIntroducer {           
     @DeclareParents(value="com.mengxiang.concert.Performance+", 
                defaultImpl = BackSinger.class)
     public static Singer singer;
}

可以发现他没有前置before,后置after或环绕round通知,只有一个 @DeclareParents ,通过她,就能将 Singer 接口的实现引入到 Performance 的实现类中,即相当于Performance 得到了新的方法。

详细解释一下:

所以很明显,我们需要写一个名为 BackSingerSinger的实现类,幕后歌者:
/** 
* 幕后歌者 
* Created by Henvealf on 2016/9/3. 
*/
public class BackSinger implements Singer{ 
      public void sing() { 
              System.out.println("嘿嘿嘿~~~哈哈哈~~~你们去哪里啊!!!"); 
      }
}
最后将 Performance 的实现dancer 与切面 SingerIntroducer 注入到容器中。
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import javax.sound.midi.Track;
/** 
* 
* Created by Henvealf on 2016/8/26. 
*/
@Configuration
@EnableAspectJAutoProxy
@ComponentScan("com.henvealf.learn.spring.concert")
public class ConcertConfig { 
        @Bean(name = "dancer") 
        public Performance dancer(){ 
                return new Dancer(); 
        } 
        @Bean 
        SingerIntroducer singerIntroducer(){ 
                return new SingerIntroducer(); 
        }
}

关于 dance 可以查看 Spring AOP 笔记一(基础概念,一个简单切面)

最后就是运行看看效果了。

import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
/** 
* 
* 
Created by Henvealf on 2016/8/27.
 */
public class Main { 
    public static void main(String[] args) {
        ApplicationContext context =
                 new AnnotationConfigApplicationContext("com.mengxiang.concert"); 
        Performance per = (Performance) context.getBean("dancer"); 
        Singer singer = (Singer)per; singer.sing(); 
        //或者这样: 
        //((Singer)per).sing(); 
    }
}

执行结果:

嘿嘿嘿~~~哈哈哈~~~你们去哪里啊!!!

我们可以发现,为了执行在Performance中引入的sing()方法,需要先将其bean转型为Singer类型才能执行sing()方法,伪装一下,不然被人发现可就闹大发了,编译器自己就不愿意。

本篇就结束喽! 谢谢大家的支持!或许还有下一篇 !

上一篇下一篇

猜你喜欢

热点阅读