第2章 Spring核心概念

Mr.Tong...
  • spring
  • spring
大约 23 分钟

概要:

  • Ioc/DI思想
  • Ioc容器
  • Bean

代码现状分析:

//业务实现
public class BookServiceImpl implements BookService{
    private BookDao bookDao = new BookDaoImpl();
    public void save() {
        bookDao.save();
    }
}
//数据层实现
public class BookDaoImpl implements BookDao{
    public void save() {
        System.out.println("book dao save...");
    }
}

上面是通过 new BookDaoImpl()来实现数据层调用,如果一旦需要更换数据层方法为BookDaoImpl2,比如:

public class BookDaoImpl2 implements BookDao{
    public void save() {
        System.out.println("book dao save..2.");
    }
}

那么上面的 new BookDaoImpl()也得改成 new BookDaoImpl()2,这样 代码耦合性偏高,解决方案如下:

  • 将对象的创建控制器有程序转移到外部,这种思想成为控制反转。

📖2.1 Ioc/DI思想

✅什么是Ioc/DI

IoC(Inversion of Control)控制反转:

使用对象时,由主动new产生对象转换为由外部提供对象,此过程中对象创建控制权由程序转移到外部,此思想称为控制反转。

spring技术对Ioc思想进行了实现:

  • 提供了一个Ioc容器,用来充当Ioc思想的"外部"。
  • Ioc容器负责对象的创建、初始化等一系列工作,被创建或被管理的对象在IoC容器中统称为Bean

DI(Dependency Injection)依赖注入:

  • 在容器中建立bean与bean之间的依赖关系的整个过程,称为依赖注入。

image-20240423225849940

这样做的目的:充分解耦

  • 使用Ioc容器管理bean(Ioc)

  • 在Ioc容器内将有依赖关系的bean进行关系绑定(DI)

最终实现的效果:

  • 使用对象时不仅可以直接从Ioc容器中获取,并且获取到的bean已经绑定了所有的依赖关系

✅什么是IOC容器

Spring创建了一个容器用来存放所创建的对象,这个容器就叫IOC容器

✅什么是Bean

spring容器中所存放的一个个对象就叫Bean或Bean对象

<!--spring容器-->
<?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.xsd">
 
   <!--bean对象-->
	<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
  
</beans>

✅Ioc入门案例(XML版)

需求分析:将BookServiceImpl和BookDaoImpl交给Spring管理,并从容器中获取对应的bean对象进行方法调用。

1.创建Maven的java项目

2.pom.xml添加Spring的依赖jar包

3.创建BookService,BookServiceImpl,BookDao和BookDaoImpl四个类

4.resources下添加spring配置文件,并完成bean的配置

5.使用Spring提供的接口完成IOC容器的创建

6.从容器中获取对象进行方法调用

步骤1:创建Maven项目

1629734010072

步骤2:添加Spring的依赖jar包

pom.xml

<dependencies>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context</artifactId>
        <version>5.2.10.RELEASE</version>
    </dependency>
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.12</version>
        <scope>test</scope>
    </dependency>
</dependencies>

步骤3:添加案例中需要的类

创建BookService,BookServiceImpl,BookDao和BookDaoImpl四个类

public interface BookDao {
    public void save();
}
public class BookDaoImpl implements BookDao {
    public void save() {
        System.out.println("book dao save ...");
    }
}
public interface BookService {
    public void save();
}
public class BookServiceImpl implements BookService {
    private BookDao bookDao = new BookDaoImpl();
    public void save() {
        System.out.println("book service save ...");
        bookDao.save();
    }
}

步骤4:添加spring配置文件

resources下添加spring配置文件applicationContext.xml,并完成bean的配置

1629734336440

步骤5:在配置文件中完成bean的配置

<?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.xsd">
 
    <!--bean标签标示配置bean
    	id属性标示给bean起名字
    	class属性表示给bean定义类型
	-->
	<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
    <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl"/>

</beans>

注意事项:bean定义时id属性在同一个上下文中(配置文件)不能重复

步骤6:获取IOC容器

使用Spring提供的接口完成IOC容器的创建,创建App类,编写main方法

public class App {
    public static void main(String[] args) {
        //获取IOC容器
		ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); 
    }
}

步骤7:从容器中获取对象进行方法调用

public class App {
    public static void main(String[] args) {
        //获取IOC容器
		ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); 
//        BookDao bookDao = (BookDao) ctx.getBean("bookDao");
//        bookDao.save();
        BookService bookService = (BookService) ctx.getBean("bookService");
        bookService.save();
    }
}

步骤8:运行程序

测试结果为:

image-20210729184337603

Spring的IOC入门案例已经完成,但是在BookServiceImpl的类中依然存在BookDaoImpl对象的new操作,它们之间的耦合度还是比较高,这块该如何解决,就需要用到下面的DI:依赖注入

✅DI入门案例(XML版)

需求:基于IOC入门案例,在BookServiceImpl类中删除new对象的方式,使用Spring的DI完成Dao层的注入

1.删除业务层中使用new的方式创建的dao对象

2.在业务层提供BookDao的setter方法

3.在配置文件中添加依赖注入的配置

4.运行程序调用方法

步骤1: 去除代码中的new

在BookServiceImpl类中,删除业务层中使用new的方式创建的dao对象

public class BookServiceImpl implements BookService {
    //删除业务层中使用new的方式创建的dao对象
    private BookDao bookDao;

    public void save() {
        System.out.println("book service save ...");
        bookDao.save();
    }
}

步骤2:为属性提供setter方法

在BookServiceImpl类中,为BookDao提供setter方法

public class BookServiceImpl implements BookService {
    //删除业务层中使用new的方式创建的dao对象
    private BookDao bookDao;

    public void save() {
        System.out.println("book service save ...");
        bookDao.save();
    }
    //提供对应的set方法
    public void setBookDao(BookDao bookDao) {
        this.bookDao = bookDao;
    }
}

步骤3:修改配置完成注入

在配置文件中添加依赖注入的配置

<?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.xsd">
    <!--bean标签标示配置bean
    	id属性标示给bean起名字
    	class属性表示给bean定义类型
	-->
    <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>

    <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
        <!--配置server与dao的关系-->
        <!--property标签表示配置当前bean的属性
        		name属性表示配置哪一个具体的属性
        		ref属性表示参照哪一个bean
		-->
        <property name="bookDao" ref="bookDao"/>
    </bean>

</beans>

注意:配置中的两个bookDao的含义是不一样的

  • name="bookDao"中bookDao的作用是让Spring的IOC容器在获取到名称后,将首字母大写,前面加set找对应的setBookDao()方法进行对象注入
  • ref="bookDao"中bookDao的作用 让Spring能在IOC容器中找到id为bookDao的Bean对象给bookService进行注入
  • 综上所述,对应关系如下:

1629736314989

步骤4:运行程序

运行,测试结果为:

image-20210729184337603

📖2.2 Ioc相关内容

✅bean基本配置

bean的基本配置主要下面四种:

属性名称功能
id定义bean的id,在容器中id值唯一
class定义bean的类型,即配置的bean的全路径类名
name定义bean的别名,可以定义多个,使用逗号(,)分号(;)空格( )分隔
scope定义bean的作用范围,默认 singleton为单例, prototype为非单例
ref指定bean,必须在容器中存在,否则会报错

下面是一个标准bean容器对象:

<beans>
   <bean id="bookService"  class="spring.service.Impl.BookServiceImpl"/>
</beans>

name属性:配置别名后可根据容器名称获取bean对象:

   <!--name:为bean指定别名,别名可以有多个,使用逗号,分号,空格进行分隔-->
    <bean id="bookService" name="service service4 bookEbi" class="com.itheima.service.impl.BookServiceImpl">
    </bean>
public class AppForName {
    public static void main(String[] args) {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
        //此处根据bean标签的id属性和name属性的任意一个值来获取bean对象
        BookService bookService = (BookService) ctx.getBean("service4");
        bookService.save();
    }
}

scope属性:如何要验证bean对象是否为单例,只需同一个bean获取两次,将对象打印到控制台,看打印出的地址值是否一致。

//配置单例
<bean id="bookDao" name="dao" class="com.itheima.dao.impl.BookDaoImpl" scope="singleton"/>
配置非单例
<bean id="bookDao" name="dao" class="com.itheima.dao.impl.BookDaoImpl" scope="prototype"/>
public class AppForScope {
    public static void main(String[] args) {
        ApplicationContext ctx = new 
        ClassPathXmlApplicationContext("applicationContext.xml");

        BookDao bookDao1 = (BookDao) ctx.getBean("bookDao");
        BookDao bookDao2 = (BookDao) ctx.getBean("bookDao");
        System.out.println(bookDao1);
        System.out.println(bookDao2);
    }
}

ref属性:bean依赖注入的ref属性指定bean,必须在容器中存在

1629771744003

如果不存在,则会报错,如下:

1629771880920

这个错误大家需要特别关注下:

1629771972886

获取bean无论是通过id还是name获取,如果无法获取到,将抛出异常NoSuchBeanDefinitionException

✅bean的实例化

Spring的IOC实例化对象的三种方式分别是:

实例化方式属性
构造方法(常用)
静态工厂(了解)factory-method:指定工厂bean配置
实例工厂(了解)
FactoryBean(实用)

🔖构造方法(常用)

1.提供可访问的构造方法

public class BookDaoImpl implements BookDao{
    public BookDaoImpl() {
        System.out.println("book dao constructor...");
    }

    public void save() {
        System.out.println("book dao save...");
    }
}

2.配置:将对应的BookDaoImpl进行bean配置

<bean id="bookDao" class="com.spring.dao.BookDaoImpl"></bean>

注意 :如果没有无参构造方法,将抛出BeanCreationException异常

3.获取bean的方法进行运行测试

public class AppForInstanceBook {
    public static void main(String[] args) {
        ApplicationContext ctx = new 
            ClassPathXmlApplicationContext("applicationContext.xml");
        BookDao bookDao = (BookDao) ctx.getBean("bookDao");
        bookDao.save();
    }
}

运行程序,如果控制台有打印构造函数中的输出,说明Spring容器在创建对象的时候也走的是构造函数

1629775972507

🔖静态工厂

1.静态工厂

public class OrderDaoFactory {
    public static OrderDao getOrderDao(){
        System.out.println("OrderDaoFactory getOrderDao runing...");
        return new OrderDaoImpl();
    }
}

2.配置bean

<bean id="orderDao" class="com.spring.factory.OrderDaoFactory" factory-method="getOrderDao"></bean>
  • class配置的是工厂类
  • 必须指定工厂中哪个方法去创建对象

3.获取bean的方法进行运行测试

public class AppForInstanceOrder {
    public static void main(String[] args) {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");

        OrderDao orderDao = (OrderDao) ctx.getBean("orderDao");

        orderDao.save();

    }
}

1629786862329

🔖实例工厂(了解)

1.一个没有静态修饰的工厂

public class UserDaoFactory {
    public UserDao getOrderDao(){
        System.out.println("UserDaoFactory getOrderDao running...");
        return new UserDaoImpl();
    }
}

2.配置

<!--实例化工厂初始化bean-->
<bean id="userDaoFactory" class="com.spring.factory.UserDaoFactory"></bean>
<bean id="userDao" factory-bean="userDaoFactory" factory-method="getOrderDao"></bean>

factory-bean指定工厂bean,factory-method指定返回实例的工厂方法

3.获取对象

public class AppUserFactory {
    public static void main(String[] args) {
     
        //spring实例化工厂创建对象
        ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
        UserDao userDao = (UserDao) ctx.getBean("userDao");
        userDao.save();
    }
}

🔖FactoryBean实例化(实用)

这第四种实例化其实是spring在实例工厂的基础上做了改进,也就是FactoryBean实例化。这种方式在Spring去整合其他框架的时候会被用到,所以这种方式需要理解掌握。

1.创建FactoryBean工厂,需要实现FactoryBean接口

public class UserDaoFactoryBean implements FactoryBean<UserDao> {

    //代替原始实例工厂中创建对象的方法
    @Override
    public UserDao getObject() throws Exception {
        return new UserDaoImpl();
    }

    @Override
    public Class<?> getObjectType() {
        return UserDao.class;
    }
}

2.配置

<bean id="userDao" class="com.spring.factory.UserDaoFactoryBean"></bean>

3.获取对象

public class AppUserFactoryBean {
    public static void main(String[] args) {
        //FactoryBean实例化
        ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
        UserDao userDao = (UserDao) ctx.getBean("userDao");
        userDao.save();
    }
}

查看源码会发现,FactoryBean接口其实会有三个方法,分别是:

T getObject() throws Exception;

Class<?> getObjectType();

default boolean isSingleton() {
		return true;
}

方法一:getObject(),被重写后,在方法中进行对象的创建并返回

方法二:getObjectType(),被重写后,主要返回的是被创建类的Class对象

方法三:没有被重写,因为它已经给了默认值,从方法名中可以看出其作用是设置对象是否为单例,默认true,从意思上来看,我们猜想默认应该是单例,如何来验证呢?

✅bean的生命周期

bean属性作用
init-method定义bean的初始化方法
destroy-method定义bean的销毁方法

🔖生命周期设置

添加生命周期的控制方法,具体的控制有两个阶段:

  • bean创建之后,想要添加内容,比如用来初始化需要用到资源
  • bean销毁之前,想要添加内容,比如用来释放用到的资源
步骤1:添加初始化和销毁方法

针对这两个阶段,我们在BooDaoImpl类中分别添加两个方法,方法名任意

public class BookDaoImpl implements BookDao {
    public void save() {
        System.out.println("book dao save ...");
    }
    //表示bean初始化对应的操作
    public void init(){
        System.out.println("init...");
    }
    //表示bean销毁前对应的操作
    public void destory(){
        System.out.println("destory...");
    }
}
步骤2:配置生命周期

在配置文件添加配置,如下:

<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl" init-method="init" destroy-method="destory"/>
步骤3:运行程序

运行AppForLifeCycle打印结果为:

1629792339889

从结果中可以看出,init方法执行了,但是destroy方法却未执行,这是为什么呢?

  • Spring的IOC容器是运行在JVM中
  • 运行main方法后,JVM启动,Spring加载配置文件生成IOC容器,从容器获取bean对象,然后调方法执行
  • main方法执行完后,JVM退出,这个时候IOC容器中的bean还没有来得及销毁就已经结束了
  • 所以没有调用对应的destroy方法

🔖close关闭容器

  • ApplicationContext中没有close方法

  • 需要将ApplicationContext更换成ClassPathXmlApplicationContext

    ClassPathXmlApplicationContext ctx = new  ClassPathXmlApplicationContext("applicationContext.xml");
    
  • 调用ctx的close()方法

    ctx.close();
    
  • 运行程序,就能执行destroy方法的内容

    1629792857608

🔖注册钩子关闭容器

  • 在容器未关闭之前,提前设置好回调函数,让JVM在退出之前回调此函数来关闭容器

  • 调用ctx的registerShutdownHook()方法

    ctx.registerShutdownHook();
    

    注意:registerShutdownHook在ApplicationContext中也没有

  • 运行后,查询打印结果

    1629792857608

两种方式介绍完后,closeregisterShutdownHook选哪个?

相同点:这两种都能用来关闭容器

不同点:close()是在调用的时候关闭,registerShutdownHook()是在JVM退出前调用关闭。

分析上面的实现过程,会发现添加初始化和销毁方法,即需要编码也需要配置,实现起来步骤比较多也比较乱。

Spring提供了两个接口来完成生命周期的控制,好处是可以不用再进行配置init-methoddestroy-method

接下来在BookServiceImpl完成这两个接口的使用:

修改BookServiceImpl类,添加两个接口InitializingBeanDisposableBean并实现接口中的两个方法afterPropertiesSetdestroy

public class BookServiceImpl implements BookService, InitializingBean, DisposableBean {
    private BookDao bookDao;
    public void setBookDao(BookDao bookDao) {
        this.bookDao = bookDao;
    }
    public void save() {
        System.out.println("book service save ...");
        bookDao.save(); 
    }
    public void destroy() throws Exception {
        System.out.println("service destroy");
    }
    public void afterPropertiesSet() throws Exception {
        System.out.println("service init");
    }
}

重新运行AppForLifeCycle类,

1629794527419

那第二种方式的实现,我们也介绍完了。

小细节

  • 对于InitializingBean接口中的afterPropertiesSet方法,翻译过来为属性设置之后

  • 对于BookServiceImpl来说,bookDao是它的一个属性

  • setBookDao方法是Spring的IOC容器为其注入属性的方法

  • 思考:afterPropertiesSet和setBookDao谁先执行?

    • 从方法名分析,猜想应该是setBookDao方法先执行

    • 验证思路,在setBookDao方法中添加一句话

      public void setBookDao(BookDao bookDao) {
              System.out.println("set .....");
              this.bookDao = bookDao;
          }
      
      
    • 重新运行AppForLifeCycle,打印结果如下:

      1629794928636

      验证的结果和我们猜想的结果是一致的,所以初始化方法会在类中属性设置之后执行。

🔖bean生命周期小结

(1)关于Spring中对bean生命周期控制提供了两种方式:

  • 在配置文件中的bean标签中添加init-methoddestroy-method属性
  • 类实现InitializingBeanDisposableBean接口,这种方式了解下即可。

(2)对于bean的生命周期控制在bean的整个生命周期中所处的位置如下:

  • 初始化容器
    • 1.创建对象(内存分配)
    • 2.执行构造方法
    • 3.执行属性注入(set操作)
    • 4.执行bean初始化方法
  • 使用bean
    • 1.执行业务操作
  • 关闭/销毁容器
    • 1.执行bean销毁方法

(3)关闭容器的两种方式:

  • ConfigurableApplicationContext是ApplicationContext的子类
    • close()方法
    • registerShutdownHook()方法

📖2.3 DI相关内容

思考:向一个类中传递数据的方式有几种?

  • 普通方法(set方法)
  • 构造方法

思考:依赖注入描述了在容器中建立bean与bean之间依赖关系的过程,如果bean运行需要的是数字或字符串呢?

  • 引用类型
  • 简单类型(基本数据类型与String)

所以依赖注入方式可以分为下面几种:

  • setter注入
    • 简单类型
    • 引用类型
  • 构造器注入
    • 简单类型
    • 引用类型

✅Setter注入

setter注入一共两步:

  • 对象需提供可访问的set方法,set的类型可以是简单数据类型或引用数据类型
  • 在配置中使用property标签,name指定set方法中的属性,ref实现引用类型注入,value实现简单数据类型注入。

🔖1.简单类型

在配置中使用property标签,value属性来实现注入简单数据类型

public class BookDaoImpl implements BookDao {
    private int connectionNum;
    private String databaseName;

    public void setConnectionNum(int connectionNum) {
        this.connectionNum = connectionNum;
    }

    public void setDatabaseName(String databaseName) {
        this.databaseName = databaseName;
    }

    public void save() {
        System.out.println("BookDao save ... connectionNum="+connectionNum+",databaseName="+databaseName);
    }
}
    <bean id="bookDao" class="com.spring.dao.Impl.BookDaoImpl" init-method="init" destroy-method="destroy" >
        <property name="databaseName" value="mysql"></property>
        <property name="connectionNum" value="100"></property>
    </bean>

🔖2.引用类型

引用类型的setter注入在上面将DI入门案例的时候写过,是在配置中使用property标签,ref属性来关联bean之间的依赖关系,实现注入引用类型

public class BookServiceImpl implements BookService {
    private BookDao bookDao;
    
    public void save() {
        System.out.println("service save");
        bookDao.save();
    }
	//set注入bookDao
    public void setBookDao(BookDaoImpl bookDao) {
        this.bookDao = bookDao;
    }
}
    <bean id="bookDao" class="com.jdk8.dao.BookDaoImpl"/>
    <bean id="bookService" class="com.jdk8.service.BookServiceImpl">
        <!--配置service与dao的关系-->
        <property name="bookDao" ref="bookDao"/>
    </bean>

✅构造器注入

构造器注入和setter注入很相似,只需将原来的set方法换成构造方法即可,也是两步:

  • 对象需提供有参的构造方法,参数可以是简单数据类型或引用数据类型
  • 在配置中使用property标签,name指定set方法中的属性,ref实现引用类型注入,value实现简单数据类型注入。

构造器参数适配(了解):

  • type属性设置按形参类型注入
  • index属性设置按形参位置注入

🔖1.简单类型

public class BookDaoImpl implements BookDao {
    private int connectionNum;
    private String databaseName;
    // 构造器注入
    public BookDaoImpl(int connectionNum, String databaseName) {
        this.connectionNum = connectionNum;
        this.databaseName = databaseName;
    }

    public void save() {
        System.out.println("BookDao save ... connectionNum="+connectionNum+",databaseName="+databaseName);
    }
    
}
    <bean id="bookDao" class="spring.dao.Impl.BookDaoImpl">
        <constructor-arg name="connectionNum" value="10"/>
        <constructor-arg name="databaseName" value="mysql"/>
    </bean>

🔖2.引用类型

public class BookServiceImpl implements BookService {
    private BookDao bookDao;
    // 构造器注入
    public BookServiceImpl(BookDao bookDao) {
        this.bookDao = bookDao;
    }

    public void save() {
        System.out.println("book service");
        bookDao.save();
    }
}
<bean id="bookDao" class="com.jdk8.dao.BookDaoImpl"/> 
<bean id="bookService" class="spring.service.Impl.BookServiceImpl">
        <constructor-arg name="bookDao" ref="bookDao"/>
</bean>

构造器参数适配(了解即可):解决配置文件与构造器形参耦合度

  • 在constructor-arg标签中使用type属性设置按形参类型注入
    <bean id="bookDao" class="spring.dao.Impl.BookDaoImpl">
        <constructor-arg type="int" value="10"/>
        <constructor-arg type="java.lang.String" value="mysql"/>
    </bean>
  • 在constructor-arg标签中使用index属性设置按形参位置注入
    <bean id="bookDao" class="spring.dao.Impl.BookDaoImpl">
        <constructor-arg index="0" value="10"/>
        <constructor-arg index="1" value="mysql"/>
    </bea

依赖注入方式选择

  • 建议使用setter注入
  • 第三方技术根据情况选择

✅集合注入

  • 数组
  • List
  • Set
  • Map
  • Properties
public class BookDaoImpl implements BookDao {
    private int[] array;
    private List<String> list;
    private Set<String> set;
    private Map<String,String> map;
    private Properties properties;
    // set
    public void setArray(int[] array) {
        this.array = array;
    }

    public void setList(List<String> list) {
        this.list = list;
    }

    public void setSet(Set<String> set) {
        this.set = set;
    }

    public void setMap(Map<String, String> map) {
        this.map = map;
    }

    public void setProperties(Properties properties) {
        this.properties = properties;
    }

    public void save() {
        System.out.println("BookDao save ... ");
        System.out.println("遍历数组:"+ Arrays.toString(array));
        System.out.println("遍历集合:"+ list);
        System.out.println("遍历set:"+ set);
        System.out.println("遍历map:"+ map);
        System.out.println("遍历properties:"+ properties);
    }

}
    <bean id="bookDao" class="com.spring.dao.Impl.BookDaoImpl">
        <!--数组-->
        <property name="array">
            <array>
                <value>100</value>
                <value>200</value>
                <value>300</value>
            </array>
        </property>
        <!--list-->
        <property name="list">
            <list>
                <value>zhang</value>
                <value>li</value>
                <value>zeng</value>
            </list>
        </property>
        <!--set-->
        <property name="set">
            <set>
                <value>zhang</value>
                <value>li</value>
                <value>li</value>
                <value>zeng</value>
            </set>
        </property>
        <!--map-->
        <property name="map">
            <map>
              <entry key="zhang" value="18"/>
              <entry key="li" value="22"/>
              <entry key="zeng" value="25"/>
            </map>
        </property>
        <!--properties-->
        <property name="properties">
            <props>
                <prop key="zhang">18</prop>
                <prop key="li">20</prop>
                <prop key="zeng">25</prop>
            </props>
        </property>
    </bean>

✅自动装配

什么是依赖自动装配?

Ioc容器根据bean所依赖的资源在容器中查找并注入到bean中的过程称为自动装配。

自动装配的方式:

  • 按类型(常用)
  • 按名称
  • 按构造方法
  • 不启用自动装配

🔖按类型(常用)

在bean中设置autoware属性为byType

public class BookServiceImpl implements BookService {
    private BookDao bookDao;
    // set注入
    public void setBookDao(BookDao bookDao) {
        this.bookDao = bookDao;
    }
    public void save() {
        System.out.println("book service");
        bookDao.save();
    }
}
<bean id="bookService" class="com.spring.service.Impl.BookServiceImpl" autowire="byType"/>

需注意两点:

  • 对象中的set方法必须有,否则将找不到
  • 如果相同类型有多个则报错,可按名称,如下
<bean id="bookDao" class="com.spring.dao.Impl.BookDaoImpl"/>
<bean id="bookDao2" class="com.spring.dao.Impl.BookDaoImpl"/>

🔖按名称

在bean中设置autoware属性为byName

<bean id="bookService" class="com.spring.service.Impl.BookServiceImpl" autowire="byName"/>

🔖依赖自动装配特征

  • 自动装配用于引用类型依赖注入,不能对简单类型进行操作
  • 使用按类型装配时(byType)必须保障容器中相同类型的bean唯一,推荐使用
  • 使用按名称装配时(byName)必须保障容器中具有指定名称的bean,因变量名与配置耦合,不推荐使用
  • 自动装配优先级低于setteri注入与构造器注入,同时出现时自动装配配置失效

📖2.4 IOC/DI配置管理第三方bean

✅数据源Druid管理

步骤1:导入druid的依赖

pom.xml中添加依赖

<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>druid</artifactId>
    <version>1.1.16</version>
</dependency>

步骤2:配置第三方bean

在applicationContext.xml配置文件中添加DruidDataSource的配置

<?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.xsd">
	<!--管理DruidDataSource对象-->
    <bean class="com.alibaba.druid.pool.DruidDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
        <property name="url" value="jdbc:mysql://localhost:3306/spring_db"/>
        <property name="username" value="root"/>
        <property name="password" value="root"/>
    </bean>
</beans>

说明:

  • driverClassName:数据库驱动
  • url:数据库连接地址
  • username:数据库连接用户名
  • password:数据库连接密码
  • 数据库连接的四要素要和自己使用的数据库信息一致。

步骤3:从IOC容器中获取对应的bean对象

public class App {
    public static void main(String[] args) {
       ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
       DataSource dataSource = (DataSource) ctx.getBean("dataSource");
       System.out.println(dataSource);
    }
}

步骤4:运行程序

打印如下结果: 说明第三方bean对象已经被spring的IOC容器进行管理

1629887733081

做完案例后,我们可以将刚才思考的两个问题答案说下:

  • 第三方的类指的是什么?

    DruidDataSource
    
  • 如何注入数据库连接四要素?

    setter注入
    

✅数据源C3P0管理

步骤1:导入C3P0的依赖

pom.xml中添加依赖

<dependency>
    <groupId>c3p0</groupId>
    <artifactId>c3p0</artifactId>
    <version>0.9.1.2</version>
</dependency>

对于新的技术,不知道具体的坐标该如何查找?

  • 直接百度搜索

  • 从mvn的仓库https://mvnrepository.com/中进行搜索

    1629888540286

步骤2:配置第三方bean

在applicationContext.xml配置文件中添加配置

<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
    <property name="driverClass" value="com.mysql.jdbc.Driver"/>
    <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/spring_db"/>
    <property name="user" value="root"/>
    <property name="password" value="root"/>
    <property name="maxPoolSize" value="1000"/>
</bean>

注意:

  • ComboPooledDataSource的属性是通过setter方式进行注入
  • 想注入属性就需要在ComboPooledDataSource类或其上层类中有提供属性对应的setter方法
  • C3P0的四个属性和Druid的四个属性是不一样的

步骤3:运行程序

程序会报错,错误如下

1629889170229

报的错为ClassNotFoundException,翻译出来是类没有发现的异常,具体的类为com.mysql.jdbc.Driver。错误的原因是缺少mysql的驱动包。

分析出错误的原因,具体的解决方案就比较简单,只需要在pom.xml把驱动包引入即可。

<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <version>5.1.47</version>
</dependency>

添加完mysql的驱动包以后,再次运行App,就可以打印出结果:

1629903845404

注意:

  • 数据连接池在配置属性的时候,除了可以注入数据库连接四要素外还可以配置很多其他的属性,具体都有哪些属性用到的时候再去查,一般配置基础的四个,其他都有自己的默认值
  • Druid和C3P0在没有导入mysql驱动包的前提下,一个没报错一个报错,说明Druid在初始化的时候没有去加载驱动,而C3P0刚好相反
  • Druid程序运行虽然没有报错,但是当调用DruidDataSource的getConnection()方法获取连接的时候,也会报找不到驱动类的错误

✅加载properties文件

🔖开启命名空间并加载properties

  • 开启context命名空间
  • 使用context命名空间,加载指定properties文件
  • 使用${}读取加载的属性值

image-20240426140550313

#jdbc.properties配置文件
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/spring_03_database
jdbc.username=root
jdbc.password=123456
<!-- 导入外部属性properties文件 -->
<context:property-placeholder location="jdbc.properties"/>
    <!-- 配置数据源,使用${}占位符读取配置文件数据 -->
    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
        <property name="driverClassName" value="${jdbc.driver}"/>
        <property name="url" value="${jdbc.url}"/>
        <property name="username" value="${jdbc.username}"/>
        <property name="password" value="${jdbc.password}"/>
    </bean>

在context标签中可以设置system-properties-mode属性来控制是否允许通过系统属性来覆盖属性占位符(Placeholder)中的值。当没有设置system-properties-mode时会优先从系统变量获取。

<context:property-placeholder location="jdbc.properties" system-properties-mode="NEVER"/>
参数作用
NEVER不允许系统属性覆盖属性占位符中的值
FALLBACK表示如果属性占位符中的值在系统属性中没有找到对应的键值,则会尝试从系统属性中获取相同键的值来替换属性占位符中的值。
OVERRIDE表示系统属性会覆盖属性占位符中的任何值,即使属性占位符中已经设置了一个值。

🔖加载properties方式

  • 不加载系统属性
 <context:property-placeholder location="jdbc.properties" system-properties-mode="NEVER"/>
  • 加载多个properties文件
<context:property-placeholder location="jdbc.properties,jdbc.properties2" system-properties-mode="NEVER"/>
  • 加载所有properties文件
<context:property-placeholder location="*.properties" />
  • 加载properties文件标准格式
<context:property-placeholder location="classpath:*.properties" />
  • 从类路径或jar包中搜索并加载properties文件
<context:property-placeholder location="classpath*:*.properties" />

📖2.5 核心容器

✅创建容器(了解)

  • 方式一:类路径加载配置文件
ApplicationContext ctx new ClassPathXmlApplicationContext("applicationContext.xml");
  • 方式二:文件路径加载配置文件
ApplicationContext ctx new FileSystemXmlApplicationContext("D:\\applicationContext.xml");
  • 加载多个配置文件
ApplicationContext ctx new ClassPathXmlApplicationContext("bean1.xml","bean2.xml");

✅获取Bean

  • 方式一:使用bean名称获取
BookDao bookDao (BookDao)ctx.getBean("bookDao");
  • 方式二:使用bean名称获取并指定类型
BookDao bookDao ctx.getBean("bookDao",BookDao.class);
  • 方式三:使用bean类型获取
BookDao bookDao ctx.getBean(BookDao.class);

✅容器类层次结构图

image-20240428203802436

✅BeanFactory初始化(了解)

  • 类路径加载配置文件
Resource resources = new ClassPathResource("applicationContext.xml");
BeanFactory bf = new XmlBeanFactory(resources);
BookDao bookDao bf.getBean("bookDao",BookDao.class);
bookDao.save();
  • BeanFactoryt创建完毕后,所有的bean均为延迟加载

✅核心容器总结

1.容器相关

  • BeanFactory是IoC容器的顶层接口,初始化BeanFactory对象时,加载的bean延迟加载
  • ApplicationContext:接口是Spring容器的核心接口,初始化时bean立即加载
  • ApplicationContext:接口提供基础的bean操作相关方法,通过其他接口扩展其功能
  • ApplicationContext接口常用初始化类
    • ClassPathXmlApplicationContext
    • FileSystemXmlApplicationContext

2.bean相关

image-20240426160310250

3.依赖注入相关

image-20240426160247386

你认为这篇文章怎么样?

  • 0
  • 0
  • 0
  • 0
  • 0
  • 0
评论
  • 按正序
  • 按倒序
  • 按热度
Powered by Waline v2.14.1