博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Spring的核心思想你了解多少?揭秘Spring依赖注入的神秘面纱
阅读量:2198 次
发布时间:2019-05-02

本文共 4882 字,大约阅读时间需要 16 分钟。

依赖注入是面向接口编程的一种体现,是Spring的核心思想。事实上依赖注入并不是什么高深的技术, 只是被Sping这么以包装就显得有些神秘。

class Main {    interface Language {        void print(String s);    }    static class Java implements Language{        @Override        public void print(String x) {            System.out.println("System.out.print(\""+ x +"\")");        }    }    static class Coder {        private Language lang = new Java();        public void helloWorld() {            lang.print("hello world");        }    }    public static void main(String[] args) {        Coder coder = new Coder();        coder.helloWorld();    }}

如上代码清单所示,Coder使用Java语言打印helloworld字符串, 在这里它不但依赖Language接口, 还依赖Java类,这使得它和Java类耦合在一起。要消除这种依赖或者说解耦很容易。

interface Language {  void print(String s);}static class Java implements Language{  @Override  public void print(String x) {    System.out.println("System.out.print(\""+ x +"\")");  }}static class Coder {  private Language lang;  public void setLang(Language lang) {    this.lang = lang;  }  public void helloWorld() {    lang.print("hello world");  }}public static void main(String[] args) {  Coder coder = new Coder();  Language java = new Java();  coder.setLang(java);  coder.helloWorld();}

我们给Coder类增加了设置具体语言的方法,使得Coder类只依赖Language接口而不依赖具体的语言实现,换言之,Coder类和具体的语言解耦了,此时我们可以轻而易举的使用其它语言代替Java,比如说使用C#。

static class CSharp implements Language{    @Override    public void print(String x) {        System.out.println("Console.Write(\""+ x +"\")");    }}public static void main(String[] args) {    Coder coder = new Coder();    Language csharp = new CSharp();    coder.setLang(csharp);    coder.helloWorld();}

这种在外部设置某个对象所依赖的具体对象的技巧就是依赖注入,这很很令人以外,一种最常见不过的编码技巧居然还有如此高大山的名称。

对于Coder类来说,确定使用何种语言原本实在编译器期确定的,使用依赖注入后,使用何种语言便延时至运行期。

Spring框架的核心思想便是基于此,不过它的实现更进一步,它把创建各个对象设置依赖关系的过程动态化和通用化了。在我们的代码清单中,创建对象和设置依赖关系的main方法只适用与当前的情况,而Spring的IOC容器能适用与任何情况

通常,Spring的依赖关系由XML表示,IOC容器解析XML完成对象的创建和依赖注入。

我们将之前的代码用Spring框架来实现:

interface Language {    void print(String s);}class Java implements Language{    @Override    public void print(String x) {        System.out.println("System.out.print(\""+ x +"\")");    }}class CSharp implements Language{    @Override    public void print(String x) {        System.out.println("Console.Write(\""+ x +"\")");    }}class Coder {    private Language lang;    public void setLang(Language lang) {        this.lang = lang;    }    public Language getLang() {        return lang;    }    public void helloWorld() {        lang.print("hello world");    }}

依赖关系将由XML配置实现

    
        
        
        
    

创建Coder对象的代码变为

public static void main(String[] args) {    ApplicationContext context = new FileSystemXmlApplicationContext("applicationContext.xml");    Coder coder = (Coder) context.getBean("coder");    coder.helloWorld();}

具体的对象创建和依赖关系的设置将由IOC根据XML配置来完成,Spring使得依赖注入机制自动化,但是依赖注入的本质却没有变花。,这篇推荐大家看下。

面向切面编程能实现不改变原有代码的前提下动态的对功能进行增强, 比如说在一个方法执行前或执行后做某些事情如记录日志、计算运行时间等等。

Spring中完美集成Aspectj,因此可以很方便的进行面向切面编程。

Spring Aspectj有几个注解用以实现常用的面向切面编程功能

@Aspectpublic class Logger {    @Before("execution(* controller.Default.*(..))")    public void before(JoinPoint join){}    @After("execution(* controller.Default.*(..))")    public void after(){}    @AfterReturning("execution(* controller.Default.*(..))")    public void afterReturning() {}    @AfterThrowing("execution(* controller.Default.*(..))")    public void afterThrowing(){}    @Around("execution(* controller.Default.*(..))")    public void around(ProceedingJoinPoint jp) {}}

如上代码所示, 此类对controller.Default类下的所有方法进行增强。

@Before注解

@Before注解修饰的方法会在被增强的方法执行前被执行

@After注解

@After注解修饰的方法会在被增强的方法执行后被执行

@AfterReturning注解

@AfterReturning注解修饰的方法会在被增强的方法执行后被执行,但前提是被修饰的方法顺利执行结束,假如方法中途抛出异常,那么AfterReturning注解修饰的方法将不会被执行,而After注解修饰的方法是无论如何都会被执行。

@AfterThrowing注解

@AfterThrowing注解修饰的方法会在被增强的方法执行出错抛出异常的情况下被执行。

@Around注解

@Around注解是@Before注解和@After注解的综合,它可以在被增强的方法的前后同时进行增强

@Around("execution(* controller.Default.*(..))")public void around(ProceedingJoinPoint jp) {    try {        System.out.println("before");        jp.proceed();        System.out.println("after");    } catch (Throwable e) {        System.out.println(e.getMessage());    }}

使用此注解,被增强的方法需要手动编码调用

jp.proceed();

如果在增强代码中不写这一句,那么被增强的方法将不会运行。

此外, 还有一个重要的注解 @Pointcut

@Pointcut注解

此注解可以用来提炼切入点

@Aspectpublic class Logger {    @Pointcut( value = "execution(* controller.Default.*(..))")    public void pointcut() {}        @Before("pointcut()")    public void before(JoinPoint join){}    @After("pointcut()")    public void after(){}    @AfterReturning("pointcut()")    public void afterReturning() {}    @AfterThrowing("pointcut()")    public void afterThrowing(){}    @Around("pointcut()")    public void around(ProceedingJoinPoint jp) {}}

@Before、@After等注解可以应用此注解声明的切入点,从而减少代码的重复。

作者:Java程序媛环环

https://blog.csdn.net/Lubanjava/article/details/100084602

往期推荐:

一只 有深度 有灵魂 的公众号0.0

你可能感兴趣的文章
【LEETCODE】147- Insertion Sort List [Python]
查看>>
【算法】- 动态规划的编织艺术
查看>>
用 TensorFlow 让你的机器人唱首原创给你听
查看>>
对比学习用 Keras 搭建 CNN RNN 等常用神经网络
查看>>
深度学习的主要应用举例
查看>>
word2vec 模型思想和代码实现
查看>>
怎样做情感分析
查看>>
用深度神经网络处理NER命名实体识别问题
查看>>
用 RNN 训练语言模型生成文本
查看>>
RNN与机器翻译
查看>>
用 Recursive Neural Networks 得到分析树
查看>>
RNN的高级应用
查看>>
TensorFlow-7-TensorBoard Embedding可视化
查看>>
轻松看懂机器学习十大常用算法
查看>>
一个框架解决几乎所有机器学习问题
查看>>
特征工程怎么做
查看>>
机器学习算法应用中常用技巧-1
查看>>
机器学习算法应用中常用技巧-2
查看>>
通过一个kaggle实例学习解决机器学习问题
查看>>
决策树的python实现
查看>>