1.IOC和DI概述

IOC(Inversion of Control):其思想是反转资源获取的方向. 传统的资源查找方式要求组件向容器发起请求查找资源. 作为回应, 容器适时的返回资源. 而应用了 IOC 之后, 则是容器主动地将资源推送给它所管理的组件, 组件所要做的仅是选择一种合适的方式来接受资源. 这种行为也被称为查找的被动形式。
DI(Dependency Injection) — IOC 的另一种表述方式:即组件以一些预先定义好的方式(例如: setter 方法)接受来自如容器的资源注入. 相对于 IOC 而言,这种表述更直接。

2.Bean的配置方式

Bean 的配置方式:通过全类名(反射)、通过工厂方法(静态工厂方法 & 实例工厂方法)、FactoryBean。

2.1 Spring通过全类名(反射)配置Bean特点:

1. 基于xml文件的方式,
 2. 通过全类名反射,
 3. 依靠IOC容器,
 4. 依赖注入的方式:属性注入,构造器注入

基于xml文件的方式配置Bean

<bean id="helloworld" class="com.java.spring.HelloWorld">
<property name="name" value="koala"></property>
</bean>

id:Bean的名称,在IOC容器中必须是惟一的。

2.2通过工厂方法配置Bean

2.2.1 通过调用静态工厂配置Bean

调用静态工厂方法创建 Bean是将对象创建的过程封装到静态方法中. 当客户端需要对象时, 只需要简单地调用静态方法, 而不用关心创建对象的细节.

要声明通过静态方法创建的 Bean, 需要在 Bean 的 class 属性里指定拥有该工厂的方法的类, 同时在 factory-method 属性里指定工厂方法的名称. 最后, 使用 <constrctor-arg> 元素为该方法传递方法参数.

示例代码:

Car.java

package com.java.spring.factory;

public class Car {
private String brand;
private double price;
public String getBrand() {
return brand;
}
public void setBrand(String brand) {
this.brand = brand;
}
public double getPrice() {
return price;
}
public void setPrice(double price) {
this.price = price;
}
public Car(String brand, double price) {
super();
this.brand = brand;
this.price = price;
}
@Override
public String toString() {
return "Car [brand=" + brand + ", price=" + price + "]";
}
}

StaticCarFactory.java

package com.java.spring.factory;
import java.util.HashMap;
import java.util.Map;
public class StaticCarFactory {
    private static Map<String,Car> cars=new HashMap<String,Car>();
    static{
        cars.put("audi",new Car("audi",300000.00));
        cars.put("ford",new Car("ford",400000.00));
    }
//静态工厂方法
    public static Car getCar(String name){
        return cars.get(name);
    }    
}

在beans-factory.xml中进行配置:

<bean id="car1" class="com.java.spring.factory.StaticCarFactory" factory-method="getCar">
<constructor-arg value="audi"></constructor-arg>
</bean>

主方法中进行实例化Bean:

package com.java.spring.factory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Main {
public static void main(String[] args){
ApplicationContext ctx=new ClassPathXmlApplicationContext("beans-factory.xml");
Car car1=(Car) ctx.getBean("car1");
System.out.println(car1);
}
}

运行后输出:

Car [brand=audi, price=300000.0]

静态工厂方法就是直接调用一个类的静态方法就能返回Bean的实例,上面代码中根据静态工厂方法public static Car getCar(String name)来配置Bean实例。

2.2.2 通过调用实例工厂方法创建 Bean

实例工厂方法: 将对象的创建过程封装到另外一个对象实例的方法里. 当客户端需要请求对象时, 只需要简单的调用该实例方法而不需要关心对象的创建细节.

要声明通过实例工厂方法创建的 Bean:

  • 在 bean 的 factory-bean 属性里指定拥有该工厂方法的 Bean;
  • 在 factory-method 属性里指定该工厂方法的名称;
  • 使用 construtor-arg 元素为工厂方法传递方法参数;

InstanceCarFactory.java

package com.java.spring.factory;
import java.util.HashMap;
import java.util.Map;
public class InstanceCarFactory {
private static Map<String,Car> cars=null;
public InstanceCarFactory() {
cars=new HashMap<String,Car>();
cars.put("Audi",new Car("Audi",300000.00));
cars.put("Ford",new Car("Ford",400000.00));
}
public Car getCar(String brand){
return cars.get(brand);
}
}

在beans-factory.xml中进行配置:

//配置工厂的实例
<bean id="carFactory" class="com.java.spring.factory.InstanceCarFactory"></bean>
//通过实例工厂方法来配置bean
<bean id="car2" factory-bean="carFactory" factory-method="getCar">
<constructor-arg value="Ford"></constructor-arg>
</bean>

2.3 实现 FactoryBean 接口在 Spring IOC 容器中配置 Bean

Spring 中有两种类型的 Bean, 一种是普通Bean, 另一种是工厂Bean, 即FactoryBean. 工厂 Bean 跟普通Bean不同, 其返回的对象不是指定类的一个实例, 其返回的是该工厂 Bean 的 getObject 方法所返回的对象。

package com.java.spring.FactoryBean;
import org.springframework.beans.factory.FactoryBean;
public class CarFactoryBean implements FactoryBean{
private String brand;
public void setBrand(String brand){
this.brand=brand;
}
@Override
//FactoryBean返回的实例
public Car getObject() throws Exception {
return new Car(brand,500000.00);
}
@Override
//FactoryBean返回的实例是否为单例
public boolean isSingleton() {
return false;
}
@Override
//FactoryBean返回的类型
public Class getObjectType() {
return Car.class;
}
}

在beans-factory.xml中进行配置:

<bean id="car3" class="com.java.spring.FactoryBean.CarFactoryBean">
<property name="brand" value="BWM"></property>
</bean>

在主方法中实例化:

Object car3= ctx.getBean("car3");

3.获取Bean

Spring提供的最常用的IOC容器实现是ApplicationContext,ApplicationContext的主要实现类ClassPathXmlApplicationContext从类路径下加载配置文件。

创建Spring的IOC对象:

ApplicationContext ctx=new ClassPathXmlApplicationContext("applicationContext.xml");

从IOC容器中获取Bean:调用ApplicationContext的getBean()方法。

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

4.Spring依赖注入的方式

4.1 属性注入

属性注入即通过 setter 方法注入Bean 的属性值或依赖的对象。使用 <property> 元素, 使用 name 属性指定 Bean 的属性名称,value 属性或 <value> 子节点指定属性值,是最常用的注入方式。

例如:

<bean id="helloworld" class="com.java.spring.HelloWorld">
<property name="name" value="koala"></property>
</bean>

4.2 构造方法注入

通过构造方法注入Bean 的属性值或依赖的对象,它保证了 Bean 实例在实例化后就可以使用。构造器注入在 <constructor-arg> 元素里声明属性, <constructor-arg> 中没有 name 属性。

Car.java

package com.java.spring;

public class Car {
private String brand;
private String corp;
private double price;
private int maxSpeed;
public Car(String brand, String corp, double price) {
super();
this.brand = brand;
this.corp = corp;
this.price = price;
}
public Car(String brand, String corp, int maxSpeed) {
super();
this.brand = brand;
this.corp = corp;
this.maxSpeed = maxSpeed;
}
@Override
public String toString() {
return "Car [brand=" + brand + ", corp=" + corp + ", price=" + price + ", maxSpeed=" + maxSpeed + "]";
}
}

4.2.1 按索引匹配入参

在XML中设置:

<bean id="car" class="com.java.spring.Car">
<constructor-arg value="Audi" index="0"></constructor-arg>
<constructor-arg value="上海" index="1"></constructor-arg>
<constructor-arg value="500000.0000" index="2"></constructor-arg>
</bean>

主方法中获取Bean实例:

package com.java.spring;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; public class Main { public static void main(String[] args){
//1.创建spring的IOC对象
ApplicationContext ctx=new ClassPathXmlApplicationContext("applicationContext.xml");
//2.从IOC容器中获取bean实例
Car car=(Car) ctx.getBean("car");
System.out.println(car);
}
}

运行后输出:

Car [brand=Audi, corp=上海, price=500000.0, maxSpeed=0]

4.2.2 按类型匹配入参

在XML中设置:

<bean id="car2" class="com.java.spring.Car">
<constructor-arg value="Audi" type="String"></constructor-arg>
<constructor-arg value="长春一汽" type="String"></constructor-arg>
<constructor-arg value="240" type="int"></constructor-arg>
</bean>

也可以使用<value></value>子节点进行赋值:

<bean id="car2" class="com.java.spring.Car">
<constructor-arg value="Audi" type="String"></constructor-arg>
<constructor-arg value="长春一汽" type="String"></constructor-arg>
<constructor-arg type="int">
<value>240</value>
</constructor-arg>
</bean>

主方法中获取Bean实例:

package com.java.spring;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; public class Main { public static void main(String[] args){
//1.创建spring的IOC对象
ApplicationContext ctx=new ClassPathXmlApplicationContext("applicationContext.xml");
//2.从IOC容器中获取bean实例
Car car=(Car) ctx.getBean("car2");
System.out.println(car);
}
}

运行后输出:

Car [brand=Audi, corp=长春一汽, price=0.0, maxSpeed=240]

5.引用其他Bean

5.1 ref引用

组成应用程序的 Bean 经常需要相互协作以完成应用程序的功能.。要使 Bean 能够相互访问, 就必须在 Bean 配置文件中指定对 Bean 的引用。在 Bean 的配置文件中, 可以通过 <ref> 元素或 ref  属性为 Bean 的属性或构造器参数指定对 Bean 的引用。

新建一个Person类Person.java

package com.java.spring;

public class Person {
private String name;
private int age;
private Car car;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public Car getCar() {
return car;
}
public void setCar(Car car) {
this.car = car;
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + ", car=" + car + "]";
}
}

在xml中进行配置:(value的值只能为字面值,即基本数据类型及其封装类,String等类型,Car非字面值,需通过ref引用其他bean)

<bean id="person" class="com.java.spring.Person">
<property name="name" value="Tom"></property>
<property name="age" value="23"></property>
<property name="car" ref="car"></property>
</bean>

主方法中获取Bean实例:

Person person=(Person) ctx.getBean("person");

运行后输出:

Person [name=Tom, age=23, car=Car [brand=Audi, corp=上海, price=500000.0, maxSpeed=0]]

5.2 内部Bean

当 Bean 实例仅仅给一个特定的属性使用时, 可以将其声明为内部 Bean。 内部 Bean 声明直接包含在 <property> 或 <constructor-arg> 元素里,不需要设置任何 id 或 name 属性。内部 Bean 不能使用在任何其他地方。

同实现上面的功能,在XML中进行配置:

<bean id="person1" class="com.java.spring.Person">
<property name="name" value="Tom"></property>
<property name="age" value="23"></property>
<property name="car">
<bean class="com.java.spring.Car">
<constructor-arg value="Audi" index="0"></constructor-arg>
<constructor-arg value="长春一汽" index="1"></constructor-arg>
<constructor-arg value="500000.0000" index="2"></constructor-arg>
</bean>
</property>
</bean>

6.级联属性赋值

和 Struts、Hiberante 等框架一样,Spring 支持级联属性的配置。

<bean id="car" class="com.java.spring.Car">
<constructor-arg value="Audi" index="0"></constructor-arg>
<constructor-arg value="上海" index="1"></constructor-arg>
<constructor-arg value="500000.0000" index="2"></constructor-arg>
</bean>
<bean id="person" class="com.java.spring.Person">
<property name="name" value="Tom"></property>
<property name="age" value="23"></property>
<property name="car" ref="car"></property>
<property name="car.price" value="300000.0000"></property>
</bean>

使用级联属性赋值的前提是car中的price必须有setPrice()方法。

运行后输出:

[name=Tom, age=23, car=Car [brand=Audi, corp=上海, price=300000.0, maxSpeed=0]]

7.集合属性的配置

在 Spring中可以通过一组内置的 xml 标签(例如: <list>, <set> 或 <map>) 来配置集合属性。

7.1 配置 java.util.List 类型的属性, 需要指定 <list>  标签, 在标签里包含一些元素。这些标签可以通过 <value> 指定简单的常量值,通过 <ref> 指定对其他 Bean 的引用。 通过<bean> 指定内置 Bean 定义。通过 <null/> 指定空元素。 甚至可以内嵌其他集合。数组的定义和 List 一样, 都使用 <list>。

示例代码:

Person.java

package com.java.spring;

import java.util.List;

public class Person {
private String name;
private int age;
private List<Car> cars;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public List<Car> getCars() {
return cars;
}
public void setCars(List<Car> cars) {
this.cars = cars;
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + ", cars=" + cars + "]";
}
}

在xml中进行配置:

<bean id="person" class="com.java.spring.Person">
<property name="name" value="Tom"></property>
<property name="age" value="23"></property>
<property name="cars">
<list>
<ref bean="car"/>
<ref bean="car2"/>
</list>
</property>
</bean>

在主方法中获取Bean实例:

Person person=(Person) ctx.getBean("person");

运行后输出:

Person [name=Tom, age=23, cars=[Car [brand=Audi, corp=上海, price=500000.0, maxSpeed=0], 
Car [brand=Audi, corp=长春一汽, price=0.0, maxSpeed=240]]]

7.2 配置 java.util.Set 需要使用 <set> 标签,定义元素的方法与 List 一样。

7.3 Java.util.Map 通过 <map> 标签定义,<map> 标签里可以使用多个 <entry> 作为子标签。每个条目包含一个键和一个值,必须在 <key> 标签里定义键。因为键和值的类型没有限制, 所以可以自由地为它们指定 <value>, <ref>, <bean> 或 <null> 元素。可以将 Map 的键和值作为 <entry> 的属性定义: 简单常量使用 key 和 value 来定义; Bean 引用通过 key-ref 和 value-ref 属性定义。

Person.java

package com.java.spring;

import java.util.Map;

public class Person {
private String name;
private int age;
private Map<String,Car> cars;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public Map<String,Car> getCars() {
return cars;
}
public void setCars(Map<String,Car> cars) {
this.cars = cars;
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + ", cars=" + cars + "]";
}
}

在xml中进行配置:

<bean id="person" class="com.java.spring.Person">
<property name="name" value="Tom"></property>
<property name="age" value="23"></property>
<property name="cars">
<map>
<entry key="A" value-ref="car"></entry>
<entry key="B" value-ref="car2"></entry>
</map>
</property>
</bean>

在主方法中获取Bean实例:

Person person=(Person) ctx.getBean("person");

运行后输出:

Person [name=Tom, age=23, cars={A=Car [brand=Audi, corp=上海, price=500000.0, maxSpeed=0],
B=Car [brand=Audi, corp=长春一汽, price=0.0, maxSpeed=240]}]

7.4 使用 <props> 定义 java.util.Properties, 该标签使用多个 <prop> 作为子标签。每个 <prop> 标签必须定义 key 属性。

DataSource.java

package com.java.spring;

import java.util.Properties;

public class DataSource {
private Properties properties;
public Properties getProperties() {
return properties;
}
public void setProperties(Properties properties) {
this.properties = properties;
}
@Override
public String toString() {
return "DataSource [properties=" + properties + "]";
}
}

在xml中进行配置:

<bean id="datasource" class="com.java.spring.DataSource">
<property name="properties">
<props>
<prop key="user">root</prop>
<prop key="password">1234</prop>
<prop key="jdbcUrl">jdbc:mysql:///test</prop>
<prop key="driverClass">com.mysql.jdbc.Driver</prop>
</props>
</property>
</bean>

在主方法中获取Bean实例:

package com.java.spring;
import java.util.Properties;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; public class Main {
public static void main(String[] args){
//1.创建spring的IOC对象
ApplicationContext ctx=new ClassPathXmlApplicationContext("applicationContext.xml");
//2.从IOC容器中获取bean实例
DataSource datasource= (DataSource) ctx.getBean(DataSource.class);
System.out.println(datasource);
}
}

运行后输出:

DataSource [properties={driverClass=com.mysql.jdbc.Driver, user=root, password=1234, jdbcUrl=jdbc:mysql:///test}]

8.使用 utility scheme 定义集合

使用基本的集合标签定义集合时, 不能将集合作为独立的 Bean 定义, 导致其他 Bean 无法引用该集合, 所以无法在不同 Bean 之间共享集合。解决这个问题的方法是使用 util schema 里的集合标签定义独立的集合 Bean. 需要注意的是, 必须在 <beans> 根元素里添加 util schema 定义。使用前需要先导入util命名空间,导入方式如下:

示例代码:

在xml中进行配置:

<util:list id="cars">
<ref bean="car"/>
<ref bean="car2"/>
</util:list>

就可以在另一个Bean里进行引用:

<bean id="person" class="com.java.spring.Person">
<property name="name" value="Tom"></property>
<property name="age" value="23"></property>
<property name="cars" ref="cars"></property>
</bean>

9.使用p命名空间

为了简化 XML 文件的配置,越来越多的 XML 文件采用属性而非子元素配置信息。Spring 从 2.5 版本开始引入了一个新的 p 命名空间,可以通过 <bean> 元素属性的方式配置 Bean 的属性。使用之前像导入util命名空间一样先导入p命名空间。这种配置方式相对于传统的配置方式更加简洁。

<bean id="person" class="com.java.spring.Person" p:age="24" p:name="Tom" p:cars-ref="cars"></bean>

wx搜索“程序员考拉”,专注java领域,一个伴你成长的公众号!

Spring中的Bean配置方式的更多相关文章

  1. spring+hibernate 配置多个数据源过程 以及 spring中数据源的配置方式

    spring+hibernate 配置多个数据源过程 以及 spring中数据源的配置方式[部分内容转载] 2018年03月27日 18:58:41 守望dfdfdf 阅读数:62更多 个人分类: 工 ...

  2. Spring 中的 Bean 配置

    内容提要 •IOC & DI 概述 •配置 bean –配置形式:基于 XML 文件的方式:基于注解的方式 –Bean 的配置方式:通过全类名(反射).通过工厂方法(静态工厂方法 & ...

  3. Spring中的Bean配置

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

  4. spring中获取bean的方式

    获取bean的方式 1.可以通过上下文的getBean方法 2.可以通过@Autowired注入 定义controller @RestController @RequestMapping(" ...

  5. Spring 中使用XML配置方式和使用注解方式实现DI

    Spring容器给我们提供了很好的环境,我们只关注主要业务即可,其他的无需关注太多.今天刚学的DI DI(Dependency Injection):依赖注入 使用XML配置文件完成依赖注入 1.1普 ...

  6. [原创]java WEB学习笔记97:Spring学习---Spring 中的 Bean 配置:IOC 和 DI

    本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...

  7. Spring中三种配置Bean的方式

    Spring中三种配置Bean的方式分别是: 基于XML的配置方式 基于注解的配置方式 基于Java类的配置方式 一.基于XML的配置 这个很简单,所以如何使用就略掉. 二.基于注解的配置 Sprin ...

  8. 【Spring】Spring中的Bean - 5、Bean的装配方式(XML、注解(Annotation)、自动装配)

    Bean的装配方式 简单记录-Java EE企业级应用开发教程(Spring+Spring MVC+MyBatis)-Spring中的Bean 文章目录 Bean的装配方式 基于XML的装配 基于注解 ...

  9. 【Spring】Spring中的Bean - 1、Baen配置

    Bean配置 简单记录-Java EE企业级应用开发教程(Spring+Spring MVC+MyBatis)-Spring中的Bean 什么是Spring中的Bean? Spring可以被看作是一个 ...

随机推荐

  1. Django 项目拆分配置文件settings.py

    使用Django命令生成一个项目的基本结构时, 配置信息默认保存在和项目目录同名的目录下的settings.py文件里, 对于一个项目而言, 这样往往是不合适的, 在实际的开发中,需要将配置文件拆分为 ...

  2. 【flask】 学习flask macro 模板

    首先 我是看着这个链接学习的 jinja2 macro官方文档 Flask开发中Macros的实践经验 Flask的Jinja2模板引擎 — 块与宏(6th) 在宏中 还可以赋值...用set标签.. ...

  3. 为什么 React Elements 会有 $$typeof 这个属性?

    简评:debug 的时候看到 element 对象中有 $$typeof 这属性,于是查了一下这到底干嘛的. 我们知道,通过 JSX 创建一个 React Elements 时: <marque ...

  4. 官宣,PyTorch 1.0 稳定版本现已推出

    简评:快来一起快乐地学习吧. 随着 PyTorch 生态系统和社区继续为开发人员提供有趣的新项目和教育资源,今天(12 月 7日)在 NeurIPS 会议上发布了 PyTorch 1.0 稳定版.研究 ...

  5. AWS 推出长期支持的 OpenJDK 免费分发版本 —— Amazon Corretto

    简评:听说 Oracle JDK 要收费了,Oracle 要限制 Java 的商业或生产用途,针对这个问题,AWS 将会推出 Amazon Corretto. Java 是 AWS 用户使用的最流行的 ...

  6. 总结day26 ----验证客户端的合法性,已经操作系统,进程的简单初识别

    前情提要 一:验证客户端的合法性: # .需要认证 # 程序和用户打交道的时候才会用到用户认证 # 对所有的客户端进行统一的认证 # # 我现在要做的事情 # 写一个server端 # 写一个clie ...

  7. Python标准库中的生成器函数

    一.用于过滤的生成器函数 - 从输入的可迭代对象中产出元素的子集,而不修改元素本身 import itertools l1 = [1,2,3,4,5] l2 = [True,False,True,Fa ...

  8. [BZOJ 5072][Lydsy1710月赛]小A的树

    传送门 \(\color{green}{solution}\) 嗯...其实我也不太会,所以大胆猜个结论吧(后来证了一下,然后放弃了...). 我们发现如果要使一个联通块的黑点数量为\(k\)的方案最 ...

  9. Focal Loss 的前向与后向公式推导

    把Focal Loss的前向和后向进行数学化描述.本文的公式可能数学公式比较多.本文尽量采用分解的方式一步一步的推倒.达到能易懂的目的. Focal Loss 前向计算 其中 是输入的数据 是输入的标 ...

  10. pytest文档博客链接

    关于pytest的博客:   https://www.cnblogs.com/yoyoketang/tag/pytest/default.html?page=2