依赖注入是面型接口编程的一种体现,是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配置实现
<?xml version="1.0" encoding="utf-8"?><beans xmlns="http://www.springframework.org/schema/beans"       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">

    <bean id="java" class="Java">    </bean>

    <bean id="csharp" class="CSharp">    </bean>

    <bean id="coder" class="Coder">        <property name="lang" ref="csharp"></property>    </bean></beans>
创建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
推荐阅读 ↓↓↓

1.不认命,从10年流水线工人,到谷歌上班的程序媛,一位湖南妹子的励志故事

2.如何才能成为优秀的架构师?

3.从零开始搭建创业公司后台技术栈

4.程序员一般可以从什么平台接私活?

5.37岁程序员被裁,120天没找到工作,无奈去小公司,结果懵了...

6.你们一直吐槽的12306,你知道他们的的架构有多牛X吗

7.滴滴业务中台构建实践,首次曝光

8.15张图看懂瞎忙和高效的区别!

9.你知道哪10大算法统治着全球吗?

一个人学习、工作很迷茫?

点击「阅读原文」加入我们的小圈子!

Spring的核心思想,这篇文章短小精悍的总结透了相关推荐

  1. Spring的核心思想,总结得非常好!

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

  2. 小马哥spring编程核心思想_Spring核心思想理解

    前言 在拉勾教育Java高薪训练营再次学习了spring,再次感叹spring的核心思想.强调一下,IOC和AOP是一个技术思想(理论),并不是spring提出的,spring在技术层次把这两个思想做 ...

  3. Spring三大核心思想

    spring三大核心思想分别是:控制反转IOC:依赖注入DI:面向切面AOP.接下来我将从是什么(what).怎么实现(how).为什么(why)来讲.只是帮助初学者理解,更具体的用法自己学习哦. 先 ...

  4. Spring三大核心思想之AOP(面向切面编程)

    Spring三大核心思想之AOP(面向切面编程) 学习Spring三大核心思想之AOP之前建议先学习:

  5. 这篇文章把数据讲透了(五):数据可视化(下)

    一.前言 上几期文章中,我们已经了解到"数据"是一个庞大的体系(如下图所示),并用了菜市场的例子,为大家讲解数据来源的含义:用买菜的例子,为大家讲解数据采集的步骤:用洗菜.择菜的例 ...

  6. Spring IoC?看这篇文章就够了...

    前言 刚开始听到 IoC,会觉得特别高大上,但其实明白原理了很简单. 跟着我的脚步,一文带你吃透 IoC 原理. 本文围绕 是何.为何.如何 来谈: 是何 上一篇文章有同学问我在官网该看哪些内容,怎么 ...

  7. 小马哥spring编程核心思想_Spring源码高级笔记之——Spring核心思想

    Spring核心思想 注意:IOC和AOP不是spring提出的,在spring之前就已经存在,只不过更偏向于理论化,spring在技术层次把这两个思想做了非常好的实现(Java) 第1节loC 1. ...

  8. Spring三大核心思想详解

    spring核心思想分三大类:控制反转(IOC),依赖注入(DI)和面向切面(AOP). 控制反转 通俗讲,控制权由应用代码中转到了外部容器,控制权的转移,是所谓反转.也就是说,正常我们都是新建对象, ...

  9. Spring三大核心思想学习笔记

    一:控制反转 **1.**Spring 三个核心思想是什么:控制反转,依赖注入和面向切面编程. **2.**Spring最核心,最基础的概念是什么?将spring类比java,java最核心,最基础的 ...

  10. 小马哥spring编程核心思想_求小马哥讲Spring栈核心编程思想 Spring IoC+Bean+Framework教程...

    这次搜集了下小马哥讲Spring栈核心编程思想 Spring IoC+Bean+Framework,最强Spring全面核心实战的视频教程,有需要的朋友可以自行下载学习. 课程简介: 小马哥出手的Sp ...

最新文章

  1. 用Python爬虫破解滑动验证码
  2. QuickPart : 用户控件包装器 for SharePoint 2007
  3. CSS3学习系列之颜色
  4. 4月21日云栖精选夜读:【校园头条】第1期:找实习、找工作时,让你脱颖而出的秘籍...
  5. Building a Simple, Local, Python Blockchain – Part 1
  6. Leet Code OJ 168. Excel Sheet Column Title [Difficulty: Easy]
  7. android linearlayout 方法,android布局----LinearLayout布局方式
  8. 第 8 天 多线程与多进程
  9. img 隐藏_仙女山美食美景色,那些隐藏在深处角色味香
  10. ribbon重试机制
  11. 台式计算机视频设备打不开,电脑视频设备被占用未能创建视频预览怎么办
  12. 第一周 从C走进C++ 002 命令行参数
  13. php post重复提交session,PHP加Session防止表单重复提交的解决方法
  14. 图片求导锐化 与模糊检测
  15. Visual Studio 2005超级集成版本下载
  16. 身份证前6位城市代码
  17. 谷歌移动UI框架Flutter入门
  18. iOS 12.0-12.1.2 完整越狱教程
  19. c语言中isupper用法,C 库函数 isupper() 使用方法及示例
  20. 高冷一字id_lol高冷而有诗意的id

热门文章

  1. Java之品优购课程讲义_day06(1)
  2. 两端对齐几种实现方案
  3. Java并发编程笔记之ArrayBlockingQueue源码分析
  4. nfs文件共享服务器搭建详解
  5. Swift---TextView用法
  6. PVD桌面怎样重新调整personal vDisks大小
  7. linux DNS 简单配置
  8. 看数据手册上的注意事项
  9. 刷leetcode不知道从哪里刷起?? 来看看,我把刷题题目顺序都排好了,每道题还带详细题解!
  10. 红警 for Mac合集(Red Alert红色警戒)