Spring | IOC技术之Bean的配置与实例化

在这里插入图片描述

👑 博主简介
   🥇 Java领域新星创作者
   🥇 阿里云开发者社区专家博主、星级博主、技术博主
🤝 交流社区BoBooY(优质编程学习笔记社区)

Bean的基础配置

1、id 与 class属性

<bean id="" class=""/>
<!--举例-->
<bean id="bookDao" class="com.bby.dao.bookDaoImpl"/>
<bean id="bookDao" class="com.bby.dao.bookDaoImpl"></bean>
  • id: bean的id,使用容器可以通过id值获取对应的bean,在一个容器中id值唯一
  • class: bean的类型,即配置的bean的全路径类名

注意:

  • class属性不能写接口如BookDao的类全名(因为接口是没办法创建对象的

  • id必须唯一,如果因为命名习惯产生分歧解决办法是设置name属性,也就是给Bean取别名

2、name属性

定义bean的别名,可定义多个,使用**逗号(,) 分号(😉 空格()**分隔

<bean id="bookDao" name="dao bookDaoImpl" class="com.bby.dao.bookDaoImpl"/>
<bean id="bookDao" name="dao,bookDaoImpl" class="com.bby.dao.bookDaoImpl"/>
<bean id="bookDao" name="dao;bookDaoImpl" class="com.bby.dao.bookDaoImpl"/>
  • 测试
  1. 这里我设置了两个别名daobookDaoImpl

在这里插入图片描述

  1. 获取这个Bean并调用它的方法
/**
 * @author BoBooY
 */
public class App {
    public static void main(String[] args) {
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");
        BookDao dao = (BookDao)applicationContext.getBean("dao");
        BookDao bookDaoImpl = (BookDao)applicationContext.getBean("bookDaoImpl");
        dao.save();
        bookDaoImpl.save();

    }
}
  1. 查看控制台结果

可以看到通过这个两个别名都可以获取到这个Bean

在这里插入图片描述

注意:

  • bean依赖注入的 ref 属性指定bean,必须在容器中存在,ref的属性值,也可也是另一个bean的name属性值,不过还是建议使用其id来进行注入

  • 如果id不存在,在获取时会获取不到,然后就会报错:NoSuchBeanDefinitionException

3、scope属性

  • singleton:单例(默认)
  • prototype:非单例
<bean id="bookDao" class="com.bby.dao.bookDaoImpl" scope="prototype"/>

分别获取单例模式和多例模式下的bean

  • 单例模式如下(singleton):

在这里插入图片描述

  • 非单例模式如下(prototype):

在这里插入图片描述


思考:

  • 为什么bean默认为单例?
    • bean为单例的意思是在Spring的IOC容器中只会有该类的一个对象
    • bean对象只有一个就避免了对象的频繁创建与销毁,达到了bean对象的复用,性能高
  • bean在容器中是单例的,会不会产生线程安全问题?
    • 如果对象是有状态对象,即该对象有成员变量可以用来存储数据的,
    • 因为所有请求线程共用一个bean对象,所以会存在线程安全问题。
    • 如果对象是无状态对象,即该对象没有成员变量没有进行数据存储的,
    • 因方法中的局部变量在方法调用完成后会被销毁,所以不会存在线程安全问题。
  • 哪些bean对象适合交给容器进行管理?
    • 表现层对象
    • 业务层对象
    • 数据层对象
    • 工具对象
  • 哪些bean对象不适合交给容器进行管理?
    • 封装实例的域对象,因为会引发线程安全问题,所以不适合。

Bean的实例化

bean本质上就是对象,对象在new的时候会使用构造方法完成,那创建bean也是使用构造方法完成的。

1、构造方法实例化

这里先说结论:Spring底层通过反射访问的类的无参构造方法来实例化Bean

验证:

  1. bookDaoImpl添加一个无参构造方法,并打印一句话,方便观察结果。
/**
 * @author BoBooY
 */
public class BookDaoImpl implements BookDao {
    public BookDaoImpl() {
        System.out.println("book dao constructor is running ....");
    }
    public void save() {
        System.out.println("book dao save ...");
    }
}

查看控制台结果:

在这里插入图片描述

从这里可以看出:Spring容器在创建对象的时候也走的是构造函数

  1. 将构造函数改成private测试
/**
 * @author BoBooY
 */
public class BookDaoImpl implements BookDao {
    private BookDaoImpl() {
        System.out.println("book dao constructor is running ....");
    }
    public void save() {
        System.out.println("book dao save ...");
    }
}

查看控制台结果:

在这里插入图片描述

从这里可以看出:Spring内部走的依然是构造函数,能访问到类中的私有构造方法,显而易见Spring底层用的是反射

  1. 使用带参构造方法测试
/**
 * @author BoBooY
 */
public class BookDaoImpl implements BookDao {
    private BookDaoImpl(int a) {
        System.out.println("book dao constructor is running ....");
    }
    public void save() {
        System.out.println("book dao save ...");
    }
}

查看控制台结果:

在这里插入图片描述

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

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

译:<init>()指定是类的构造方法,即该类的无参构造方法

  • Caused by: 引发
  • BeanInstantiationException:翻译为bean实例化异常
  • No default constructor found:没有一个默认的构造函数被发现

2、静态工厂实例化

  1. 准备一个OrderDao和OrderDaoImpl类

    public interface OrderDao {
        public void save();
    }
    
    public class OrderDaoImpl implements OrderDao {
        public void save() {
            System.out.println("order dao save ...");
        }
    }
    
  2. 创建一个工厂类OrderDaoFactory并提供一个静态方法

    //工厂类
    public class OrderDaoFactory {
        public static OrderDao getOrderDao(){
            return new OrderDaoImpl();
        }
    }
    
  3. 在spring的配置文件applicationContext.xml中添加以下内容

    <bean id="orderDao" class="com.bby.factory.OrderDaoFactory" factory-method="getOrderDao"/>
    
    • class:工厂类的类全名
    • factory-mehod:具体工厂类中创建对象的方法名
  4. 获取Bean测试

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

在这里插入图片描述

3、实例工厂实例化

  1. 创建一个工厂类OrderDaoFactory2并提供一个普通方法,注意此处和静态工厂的工厂类不一样的地方是方法不是静态方法

    //实例工厂
    public class OrderDaoFactory2 {
        public OrderDao getOrderDao(){
            return new OrderDaoImpl();
        }
    }
    
  2. 在spring的配置文件中添加以下内容

    <bean id="orderDaoFactory2" class="com.bby.factory.OrderDaoFactory2"/>
    <bean id="orderDao" factory-method="getOrderDao" factory-bean="orderDaoFactory2"/>
    
    • factory-bean:工厂的实例对象

    • factory-method:工厂对象中的具体创建对象的方法名

  3. 获取Bean进行测试

    /**
     * @author BoBooY
     */
    public class App {
        public static void main(String[] args) {
            ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
            OrderDao orderDao = (OrderDao) ctx.getBean("orderDao2");
            orderDao.save();
    
        }
    }
    

在这里插入图片描述

实例化工厂运行的顺序是:

  • 创建实例化工厂对象,对应的是第一行配置

  • 调用对象中的方法来创建bean

🚩 FactoryBean的使用

实例工厂实例化的方式配置的过程还是比较复杂,所以Spring为了简化这种配置方式就提供了一种叫FactoryBean的方式来简化开发。

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

    /**
     * @author BoBooY
     */
    public class OrderDaoFactoryBean implements FactoryBean<OrderDao> {
        //代替原始实例工厂中创建对象的方法
        public OrderDao getObject() throws Exception {
            return new OrderDaoImpl();
        }
        //返回所创建类的Class对象
        public Class<?> getObjectType() {
            return OrderDao.class;
        }
    }
    
  2. 在Spring的配置文件中进行配置

    <bean id="orderDao" class="com.bby.factory.OrderDaoFactoryBean"/>
    
  3. 获取Bean进行测试

在这里插入图片描述


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

T getObject() throws Exception;

Class<?> getObjectType();

default boolean isSingleton() {
		return true;
}
  • getObject():被重写后,在方法中进行对象的创建并返回
  • getObjectType():被重写后,主要返回的是被创建类的Class对象
  • isSingleton:没有被重写,因为它已经给了默认值,从方法名中可以看出其作用是设置对象是否为单例,默认true

尾言:创作不易,如果本文的内容对您有帮助,还望客官可以支持一下博主,👍(点赞)+✏️(评论)+⭐️(收藏)是我创作的巨大动力!

  • 108
    点赞
  • 106
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 197
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 197
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

-BoBooY-

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值