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

class Main {interface Language {void print(String s);}static class Java implements Language{@Overridepublic 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{@Overridepublic 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{@Overridepublic 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{@Overridepublic void print(String x) {System.out.println("System.out.print(\""+ x +"\")");}
}
class CSharp implements Language{@Overridepublic 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/beanshttp://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有几个注解用以实现常用的面向切面编程功能

@Aspect
public 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注解

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

@Aspect
public 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

Spring的核心思想,总结得非常好!相关推荐

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

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

  2. Spring三大核心思想

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

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

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

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

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

  5. Spring三大核心思想详解

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

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

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

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

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

  8. Spring框架核心思想

    Spring两个核心解决思想方案解析 IOC (Inverstion Object Control)控制反转 / DI (Dependency Injected) 依赖注入 AOP(Aspect Or ...

  9. 小马哥spring编程核心思想_小马哥讲Spring核心编程思想

    小马哥讲Spring核心编程思想 ├─第01章:Spring Framework总览 (12讲) │      01丨课程介绍.mp4 │      02丨内容综述.mp4 │      03丨课前准 ...

最新文章

  1. IBM 揭晓全球第一项 2纳米芯片技术,为半导体领域实现重大突破
  2. Myeclipse 10.5 下载地址
  3. 印度威普罗集团斥资5 亿美元收购云计算解决方案供应商 Appirio
  4. LeetCode Contains Duplicate (判断重复元素)
  5. Maven之自定义archetype生成项目骨架(一)
  6. 反汇编基础-数据类型以及C++引用、指针反汇编后的概括
  7. 十行代码实现十亿图片检索,我们把它开源了
  8. Maven Web项目配置Mybatis出现SqlSessionFactory错误的解决方案
  9. mcafee迈克菲官网由于验证缺陷,导致暴力破解到企业授权号获取到所有企业用户的Grant number
  10. 毕业生写论文必备!!超详细讲解参考文献格式
  11. IT服务及相关概念界定
  12. Web Moudle
  13. 读《拆掉思维的墙》小记
  14. arduino旋转编码器控制步进电机
  15. 简单易懂 LNMP 架构详解适合入门级别可跟做
  16. (一)框架搭建,前端路由设置,自定义寻找指定路径(Django+Vue+Mysql,数据库管理数据分析网站)
  17. 计算机EI会议论文,和EI期刊论文有什么区别? - 易智编译EaseEditing
  18. 基于Vue的医院内部管理系统(医生、患者、挂号、药房)文档+答辩PPt+项目源码+演示视频
  19. Chrome 浏览器最牛插件之一 Vimium
  20. 九种js弹出对话框的方法

热门文章

  1. 鸟哥的Linux私房菜(基础篇)- 第七章、Linux 文件与目录管理
  2. webpack 之 code spliting
  3. 「镁客早报」第九城市与法拉第未来成立合资公司,出资6亿美元;华为获得韩国运营商5G订单,占比95%...
  4. Docker技术笔记:Docker入门浅尝
  5. MYSQL如何导出存储过程和触发器?
  6. Entity Framework Unit Testing problem and solution(转)
  7. AppCode下的cs类 取得相关路径
  8. Java-JSTL(JSP标准标签库)
  9. VTK 7.0 安装中内存读取冲突 no override found for “.. 解决记录
  10. Linux线程同步之条件变量