当前位置: 首页 > news >正文

【spring】第二篇 bean实例化

对象已经能交给Spring的IOC容器来创建了,但是容器是如何来创建对象的呢?

就需要研究下bean的实例化过程,在这块内容中主要解决两部分内容,分别是

  • bean是如何创建的

  • 实例化bean的三种方式,构造方法,静态工厂实例工厂

在讲解这三种创建方式之前,我们需要先确认一件事:bean本质上就是对象,对象在new的时候会使用构造方法完成,那创建bean也是使用构造方法完成的。

基于这个知识点出发,我们来验证spring中bean的三种创建方式。

目录

一、构造方法实例化

优化

二、静态工厂实例化

使用工厂来创建对象

优化

三、实例工厂实例化

优化:FactoryBean使用

小结


一、构造方法实例化

优化

(1)准备需要被创建的类

准备一个BookDao和BookDaoImpl类,

package com.water.dao;public interface BookDao {public void save();
}

在BookDaoImpl类中添加一个无参构造函数,并打印一句话,方便观察结果。 

package com.water.dao.impl;import com.water.dao.BookDao;public class BookDaoImpl implements BookDao {public BookDaoImpl() {System.out.println("book dao constructor is running ....");}public void save() {System.out.println("book dao save ...");}
}

(2)将类配置到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标签标示配置beanid属性标示给bean起名字class属性表示给bean定义类型--><bean id="bookDao" class="com.water.dao.impl.BookDaoImpl"></bean>
</beans>

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

(4)将构造函数改成private测试

运行程序,能执行成功,

说明内部走的依然是构造函数,能访问到类中的私有构造方法,显而易见Spring底层用的是反射。

(5)构造函数中添加一个参数测试

运行程序,程序会报错,说明Spring底层使用的是类的无参构造方法。

接下来,我们主要研究下Spring的报错信息,来学一学如阅读。

  • 错误信息从下往上依次查看,因为上面的错误大都是对下面错误的一个包装,最核心错误是在最下面

  • Caused by: java.lang.NoSuchMethodException: com.itheima.dao.impl.BookDaoImpl.<init>()

    • Caused by 翻译为引起,即出现错误的原因

    • java.lang.NoSuchMethodException:抛出的异常为没有这样的方法异常

    • com.itheima.dao.impl.BookDaoImpl.<init>():哪个类的哪个方法没有被找到导致的异常,<init>()指定是类的构造方法,即该类的无参构造方法

如果最后一行错误获取不到错误信息,接下来查看第二层:

Caused by: org.springframework.beans.BeanInstantiationException: Failed to instantiate [com.itheima.dao.impl.BookDaoImpl]: No default constructor found; nested exception is java.lang.NoSuchMethodException: com.itheima.dao.impl.BookDaoImpl.<init>()

  • nested:嵌套的意思,后面的异常内容和最底层的异常是一致的

  • Caused by: org.springframework.beans.BeanInstantiationException: Failed to instantiate [com.itheima.dao.impl.BookDaoImpl]: No default constructor found;

    • Caused by: 引发

    • BeanInstantiationException:翻译为bean实例化异常

    • No default constructor found:没有一个默认的构造函数被发现

看到这其实错误已经比较明显,

Exception in thread "main" org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'bookDao' defined in class path resource [applicationContext.xml]: Instantiation of bean failed; nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [com.itheima.dao.impl.BookDaoImpl]: No default constructor found; nested exception is java.lang.NoSuchMethodException: com.itheima.dao.impl.BookDaoImpl.<init>()。

至此,关于Spring的构造方法实例化就已经学习完了,因为每一个类默认都会提供一个无参构造函数,所以其实真正在使用这种方式的时候,我们什么也不需要做。这也是我们以后比较常用的一种方式。

二、静态工厂实例化

使用工厂来创建对象

在讲这种方式之前,我们需要先回顾一个知识点是使用工厂来创建对象的方式:

  1. 准备一个OrderDao和OrderDaoImpl类
  2. 创建一个工厂类OrderDaoFactory并提供一个静态方法
  3. 编写AppForInstanceOrder运行类,在类中通过工厂获取对象
public interface OrderDao {public void save();
}public class OrderDaoImpl implements OrderDao {public void save() {System.out.println("order dao save ...");}
}
//静态工厂创建对象
public class OrderDaoFactory {public static OrderDao getOrderDao(){return new OrderDaoImpl();}
}
public class AppForInstanceOrder {public static void main(String[] args) {//通过静态工厂创建对象OrderDao orderDao = OrderDaoFactory.getOrderDao();orderDao.save();}
}

如果代码中对象是通过上面的这种方式来创建的,如何将其交给Spring来管理呢?

优化

这就要用到Spring中的静态工厂实例化的知识了,具体实现步骤为:

(1)在spring的配置文件application.properties中添加以下内容,

<?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标签标示配置beanid属性标示给bean起名字class属性表示给bean定义类型 工厂类的类全名factory-mehod:具体工厂类中创建对象的方法名--><bean id="orderDao" class="com.water.dao.factory.OrderDaoFactory" factory-method="getOrderDao"/>
</beans>

(3)在AppForInstanceOrder运行类,使用从IOC容器中获取bean的方法进行运行测试,

package com.water;import com.water.dao.BookDao;
import com.water.dao.OrderDao;
import com.water.dao.factory.OrderDaoFactory;
import com.water.server.BookService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;public class App {public static void main( String[] args ) {ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");OrderDao orderDao = (OrderDao) ctx.getBean("orderDao");orderDao.save();}
}

运行后,可以查看到结果, 

这样的好处就是,在工厂的静态方法中,我们除了new对象还可以做其他的一些业务操作,这些操作必不可少。

public class OrderDaoFactory {public static OrderDao getOrderDao(){System.out.println("factory setup....");//模拟必要的业务操作return new OrderDaoImpl();}
}

 之前new对象的那种方式就无法添加其他的业务内容。

三、实例工厂实例化

package com.water.dao;public interface OrderDao {public void save();
}

 

package com.water.dao.impl;import com.water.dao.OrderDao;public class OrderDaoImpl implements OrderDao {public void save() {System.out.println("order dao save ...");}
}
package com.water;import com.water.dao.BookDao;
import com.water.dao.OrderDao;
import com.water.dao.factory.OrderDaoFactory;
import com.water.server.BookService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;public class App {public static void main( String[] args ) {//创建实例工厂对象OrderDaoFactory OrderDaoFactory = new OrderDaoFactory();//通过实例工厂对象创建对象OrderDao OrderDao = OrderDaoFactory.getOrderDao();OrderDao.save();}
}

对于上面这种实例工厂的方式如何交给Spring管理呢?

优化:FactoryBean使用

(1)创建一个OrderDaoFactoryBean的类,实现FactoryBean接口,重写接口的方法

FactoryBean接口其实会有三个方法,分别是:

T getObject() throws Exception;Class<?> getObjectType();default boolean isSingleton() {return true;
}

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

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

方法三:没有被重写,因为它已经给了默认值,从方法名中可以看出其作用是设置对象是否为单例,默认true,单例。

package com.water.dao.factory;import com.water.dao.OrderDao;
import com.water.dao.impl.OrderDaoImpl;
import org.springframework.beans.factory.FactoryBean;public class OrderDaoFactoryBean implements FactoryBean<OrderDao> {@Overridepublic OrderDao getObject() throws Exception {return new OrderDaoImpl();}@Overridepublic Class<?> getObjectType() {return OrderDao.class;}public boolean isSingleton() {return false;}
}

(2)在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标签标示配置beanid属性标示给bean起名字class属性表示给bean定义类型 工厂类的类全名factory-mehod:具体工厂类中创建对象的方法名--><bean id="orderDao" class="com.water.dao.factory.OrderDaoFactoryBean"/>
</beans>

(3)运行类不用做任何修改,直接运行,

package com.water;import com.water.dao.BookDao;
import com.water.dao.OrderDao;
import com.water.dao.factory.OrderDaoFactory;
import com.water.server.BookService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;public class App {public static void main( String[] args ) {//创建实例工厂对象ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");OrderDao orderDao1 = (OrderDao) ctx.getBean("orderDao");System.out.println(orderDao1);OrderDao orderDao2 = (OrderDao) ctx.getBean("orderDao");System.out.println(orderDao2);}
}

通过验证,会发现默认是单例,那如果想改成单例具体如何实现?只需要将isSingleton()方法进行重写,修改返回为false即可。

重新运行,查看结果,

从结果中可以看出现在已经是非单例了,但是一般情况下我们都会采用单例,也就是采用默认即可。所以isSingleton()方法一般不需要进行重写。

小结

通过这一节的学习,需要掌握:

(1)bean是如何创建的呢?

构造方法

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

  • 构造方法(常用)

  • 静态工厂(了解)

  • 实例工厂(了解)

    • FactoryBean(实用)

这些方式中,重点掌握构造方法FactoryBean即可。

需要注意的一点是,构造方法在类中默认会提供,但是如果重写了构造方法,默认的就会消失,在使用的过程中需要注意,如果需要重写构造方法,最好把默认的构造方法也重写下。

http://www.lryc.cn/news/360814.html

相关文章:

  • MVC和MVVM
  • 【康耐视国产案例】智能AI相机机器视觉精准快速实现包裹标签的智能粘贴
  • 发现真正的诉求
  • Spring Boot配置MySQL数据库连接数
  • springboot595基于Java的大学生迎新系统-手把手调试搭建
  • 20 道大模型面试问题(含答案)
  • 【Java面试】四、MySQL篇(上)
  • 【Python】collections模块:高效处理数据的利器
  • Vue3实战笔记(51)—Vue 3封装带均线的k线图
  • 信息与未来2015真题笔记
  • 【成功解决】Access token invalid or no longer valid
  • 【Bug】修改计算机名称出现ip无法连接mysql数据库
  • 米尔MYC-Y6ULX-V2开发板测评记录
  • 装修全流程
  • 探索微软Edge
  • Java面试——专业技能
  • C#按钮样式设置XMAL
  • EmmyLua注释详解
  • Linux内核 -- 启用 Linux 内核调试信息
  • vs2019 无法打开QT的UI文件
  • Python | A + B问题|||
  • JRT连接希森美康出图
  • UVa11604 General Sultan
  • USB - ACK、NAK和STALL的含义
  • 查看 WSL2 (Windows Subsystem for Linux 2) IP 地址
  • 如何判断一个JavaScript对象是否为空?
  • 小白跟做江科大32单片机之LED闪烁
  • “世界酒中国菜”系列活动如何助推乡村振兴和文化交流?
  • 上位机图像处理和嵌入式模块部署(f407 mcu中fatfs中间件使用)
  • LeetCode/NowCoder-栈和队列OJ练习