IOC和DI

网上概念很多,感兴趣可以去搜一搜,在这里我就给个比喻:

IOC:以前我们买东西都要去商店买,用了IOC之后,我们只要在门口放个箱子, Spring就会给我相应商品,ಠᴗಠ

举个例子

  1. class A{}  
  2.     
  3. class B{  
  4.     private A a;  
  5.     public void setA(A a){  
  6.          this.a=a;  
  7.     }  
  8. }  

在传统的写法中,在创建A B对象,他们之间的关联需要我们手动设置,b.set(a),而在使用Spring之后,这一步就不需要了,由框架为我们设置相关内容

IOC的发展:

  1. 分离接口与实现
  2. 工厂设计模式
  3. 翻转控制

基于XML配置bean

<bean>中的class属性填入bean的全类名,通过反射方式在IOC容器中创建Bean,所以要求Bean中必须有无参的构造函数。

属性id用来表示容器中的id,id唯一

ApplicationContext代表IOC容器,实际这是一个接口。在SpringIOC容器读取Bean配置创建Bean实例之前,必须对它进行实例化,只有在容器实例化后,才可以从IOC容器里获取Bean实例并使用。

Spring为我们提供了两种类型的IOC容器实现:

  1. BeanFactory:IOC容器的基本实现
  2. ApplicationContext:提供了更多高级特性,是BeanFactory的子接口
  3. ClassPathXmlApplicationContext是ApplicationContext的实现类

BeanFactory是Spring的基础设施,面向Spring本身,ApplicationContext面向Spring框架开者,几乎所有应用场合都直接使用ApplicationContext而非底层BeanFactory,当然,无论使用何种方式,配置文件都相同。

ApplicationContext的主要实现类:

ClassPathXmlApplicationContext:从类路径下加载配置文件

FileSystemXmlApplicationContext:从文件系统中加载配置文件

ConfigurableApplicationContext扩展于ApplicationContext,新增两个主要方法:refresh()和close(),让ApplicationContext具有启动/刷新和关闭上下文的功能。

ApplicationContext在初始化上下文时就实例化所有单例的Bean。

WebApplication是专门为WEB应用而准备的,它允许从相对于WEB根目录的路径中完成初始化工作。

接下来说说getBean()这个方法,getBean()这个方法实际是存在在BeanFactory这个接口中,我们看看下面这张截图:

非常多个重载,我们在之前的例子中也可以这么写,利用类型返回IOC容器:

HelloWorld helloWorld = ctx.getBean(HelloWorld.class);

也是没问题的,这样有缺点吗?

加入我在配置文件中创建了两个<bean>并且都是HelloWorld的JavaBean,这样Spring就识别不出是哪一个<bean>了,因此不推荐这么写,推荐通过id名返回IOC容器,创建对象

HelloWorld helloWorld = (HelloWorld) ctx.getBean("helloWorld");

依赖注入

  1. 属性注入
  2. 构造器注入
  3. 工厂方法注入(很少使用,不推荐)

先说说最常用的属性注入,这个属性注入就是我们先前HelloWorld中使用的方法,通过setter方法注入Bean的属性值或依赖的对象,通过<property>标签,使用name指定Bean的属性名称,value属性指定属性值

<property name="name" value="World"></property>

接着说构造方法注入,通过构造方法注入Bean的属性值或依赖对象,它保证了Bean实例在实例化后就可以使用。

构造器注入需要用<constructor-arg>元素里声明属性,注意<constructor-arg>中没有name属性

我们新创建Car类作为测试:

  1. public class Car {  
  2.     private String brand;  
  3.     private String corp;  
  4.     private Integer price;  
  5.     private Integer maxSpeed;  
  6.     
  7.     public Car(String brand, String corp, Integer price) {  
  8.         this.brand = brand;  
  9.         this.corp = corp;  
  10.         this.price = price;  
  11.     }  
  12.     
  13.     @Override  
  14.     public String toString() {  
  15.         return "Car{" +  
  16.                 "brand='" + brand + '\'' +  
  17.                 ", corp='" + corp + '\'' +  
  18.                 ", price=" + price +  
  19.                 ", maxSpeed=" + maxSpeed +  
  20.                 '}';  
  21.     }  
  22. }  

这里故意在构造器里少放一个字段,来到配置文件

  1. <bean id="car" class="com.figsprite.bean.Car">  
  2.     <constructor-arg value="奥迪"></constructor-arg>  
  3.     <constructor-arg value="上海"/>  
  4.     <constructor-arg value="3000000"/>  
  5. </bean>  

在Main中输入测试代码:

  1. Car car = (Car)ctx.getBean("car");  
  2. System.out.println(car.toString());  

看结果,很有意思的是我已经将HelloWorld.java相关的代码注释了,可是还是出现了我们上一讲所写的注释,这就应证了之前所说的:

ApplicationContext在初始化上下文时就实例化所有单例的Bean。

接下来再配置一个构造器:

  1. public Car(String brand, String corp, int price, int maxSpeed) {  
  2.     this.brand = brand;  
  3.     this.corp = corp;  
  4.     this.price = price;  
  5.     this.maxSpeed = maxSpeed;  
  6. }  

    来到配置文件:

  7. <bean id="car2" class="com.figsprite.bean.Car">  
  8.     <constructor-arg value="海南马自达"></constructor-arg>  
  9.     <constructor-arg value="上海"/>  
  10.     <constructor-arg value="3000000"/>  
  11.     <constructor-arg value="78"/>  
  12. </bean>  
  1. Car car2 = (Car)ctx.getBean("car2");  
  2. System.out.println(car2.toString());  

发现没什么问题,但如果构造器是品牌名、场地、最大速度,Spring能识别出来吗?

先将price改成double,否则不满足重载规则,写上新的配置文件

  1. <bean id="car2" class="com.figsprite.bean.Car">  
  2.     <constructor-arg value="海南马自达"></constructor-arg>  
  3.     <constructor-arg value="上海"/>  
  4.     <constructor-arg value="78"/>  
  5. </bean>  

发现并没有得到我们想要的,如何解决这个问题呢?

此时需要用到type属性,

  1. <bean id="car" class="com.figsprite.bean.Car">  
  2.     <constructor-arg value="奥迪" type="java.lang.String"></constructor-arg>  
  3.     <constructor-arg value="上海" type="java.lang.String"/>  
  4.     <constructor-arg value="3000000" type="double"/>  
  5. </bean>  
  6.     
  7. <bean id="car2" class="com.figsprite.bean.Car">  
  8. <constructor-arg value="海南马自达" type="java.lang.String"></constructor-arg>  
  9. <constructor-arg value="上海" type="java.lang.String"/>  
  10. <constructor-arg value="78" type="int"/>  
  11. </bean>  

嗯,是我们想要的了,使用构造器注入属性值可以指定参数的位置和参数类型,这里还有一个index的属性,用于不按构造器参数顺序写标签时使用。大家可以自行尝试一下

Spring的Bean配置的更多相关文章

  1. 【转】Spring学习---Bean配置的三种方式(XML、注解、Java类)介绍与对比

    [原文]https://www.toutiao.com/i6594205115605844493/ Spring学习Bean配置的三种方式(XML.注解.Java类)介绍与对比 本文将详细介绍Spri ...

  2. spring中bean配置和bean注入

    1 bean与spring容器的关系 Bean配置信息定义了Bean的实现及依赖关系,Spring容器根据各种形式的Bean配置信息在容器内部建立Bean定义注册表,然后根据注册表加载.实例化Bean ...

  3. spring中bean配置和注入场景分析

    bean与spring容器的关系 Bean配置信息定义了Bean的实现及依赖关系,Spring容器根据各种形式的Bean配置信息在容器内部建立Bean定义注册表,然后根据注册表加载.实例化Bean,并 ...

  4. Spring 梳理-bean配置与装配

    1       bean配置与装配 1.1      bean在XML文件中进行显示配置并装配 1.2      bean在JavaConfig中显示配置并装配 1.2.1  优点:类型是安全的,编译 ...

  5. 关于spring中bean配置的几件小事

    一.IOC和DI 1.IOC(Inversion of Control) 其思想是反转资源获取的方向.传统的资源查找方式要求组件向容器发起请求查找资源,作为回应,容器适时的返回资源:而应用了IOC之后 ...

  6. 使用@Configuration注解来代替Spring的bean配置

    下面是一个典型的Spring配置文件(application-config.xml): <beans> <bean id="orderService" class ...

  7. Spring中bean的配置

    先从IOC说起,这个概念其实是从我们平常new一个对象的对立面来说的,我们平常使用对象的时候,一般都是直接使用关键字类new一个对象,那这样有什么坏处呢?其实很显然的,使用new那么就表示当前模块已经 ...

  8. Spring之Bean的配置方式

    在博客中为了演示容器Bean实例化时暴露出的几个接口,将UserBean配置在XML中,其实常见的Bean的配置有3种.1.基于xml配置Bean 2.使用注解定义Bean 3.基于java类提供Be ...

  9. spring中bean的配置详解--定义parent

    在工作中碰到了好多的配置文件,具体来说是spring 中bean配置的parent的配置,搞的我一头雾水,仔细看一下spring中有关bean的配置,剖析一下,具体什么含义! 一.Spring IoC ...

随机推荐

  1. C#语言のC#扩展方法(.Net特性)

    this在C#中的常见用法:1.在C#中,this关键字代表当前实例,我们可以用this.来调用当前实例的成员方法,变量,属性,字段等; 2.也可以用this来做为参数状当前实例做为参数传入方法. 3 ...

  2. flask基础入门

    Flask是一个基于Python开发并且依赖jinja2模板和Werkzeug WSGI服务的一个微型框架,对于Werkzeug本质是Socket服务端,其用于接收http请求并对请求进行预处理,然后 ...

  3. No such property: FOR_RUNTIME for class: org.gradle.api.attributes.Usage

    自从Android studio升级到3.1版本,谷歌又折腾出让你意想不到的错误.... 从github上下了个项目用来学习,却出现了如下错误: No such property: FOR_RUNTI ...

  4. 【转】OS X Base System 上没有足够的空间来进行安装

    今天在windows环境下安装IOS虚拟机,安装过程中报了一个错:“OS X Base System ”上没有足够的空间来进行安装.如图: 之后的解决办法是:点击上方的[实用工具]->[磁盘工具 ...

  5. ldconfig 的简单用法

    ldconfig: configure dynamic linker run-time binds. 比如,我编译完 boost 后(boost 的有些库需要编译后生成动态或静态库使用,例如 Pyth ...

  6. 接口测试,获取登录后的cookies

    参见: http://www.cnblogs.com/testwang/p/6023394.html

  7. [MicroPython]STM32F407开发板DIY声光控开关

    1.实验目的 1. 学习在PC机系统中扩展简单I/O 接口的方法. 2. 进一步学习编制数据输出程序的设计方法. 3. 学习光敏模块的工作原理. 4. 学习声音的工作原理. 5. 学习F40 7Mic ...

  8. B+Tree原理及mysql的索引分析

    一.索引的本质 MySQL官方对索引的定义为:索引(Index)是帮助MySQL高效获取数据的数据结构.提取句子主干,就可以得到索引的本质:索引是数据结构. 我们知道,数据库查询是数据库的最主要功能之 ...

  9. BZOJ1390 CEOI2008 Fences 凸包、Floyd最小环/DP

    传送门 为了方便描述把固定点叫做白色点,Tree叫做黑色点 一种基于特殊性质的做法: 如果不算入选白色的权值,那么一定会选中所有白色点构成的凸包上的点,因为能够尽可能围更多的黑色点.然后我们在这个基础 ...

  10. 容器技术|Docker三剑客之docker-compose

    三剑客简介 docker-machine docker技术是基于Linux内核的cgroup技术实现的,那么问题来了,在非Linux平台上是否就不能使用docker技术了呢?答案是可以的,不过显然需要 ...