【Spring】bean的实例化

这里写自定义目录标题

  • [1. 构造方法](#1. 构造方法)
  • [2. 静态工厂](#2. 静态工厂)
  • [3. 实例工厂](#3. 实例工厂)
  • [4. FactoryBean(实用)](#4. FactoryBean(实用))

1. 构造方法

实例化bean默认调用无参的构造方法

这种方法只需要在Spring config文件中配置(以bookDao为例):

xml 复制代码
<bean id="bookDao" name="dao" class="com.example.demo231116.dao.impl.BookDaoImpl" />

2. 静态工厂

工厂类实现:

java 复制代码
package com.example.demo231116.factory;

import com.example.demo231116.dao.BookDao;
import com.example.demo231116.dao.impl.BookDaoImpl;

public class BookDaoFactory {
    public static BookDao getBookDao(){
        System.out.println("Factory method....");
        return new BookDaoImpl();
    }

想要让bean调用工厂类,如下配置:

xml 复制代码
<bean id="bookDaoFactory" class="com.example.demo231116.factory.BookDaoFactory" factory-method="getBookDao" />

id 为从配置中调用的名称,class 是调用到的类,我们想要返回的不是这个工厂类,而是这个工厂类中具体的getBookDao方法里返回的对象,那么就要进一步定义factory-method,指定为具体的方法名称。

然后使用bean调用的时候:

java 复制代码
public class Demo231116Application2 {
    public static void main(String[] args) {
        // 获取IoC容器
        ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
        // 获取bean
        BookDao bookDao2 = (BookDao) ctx.getBean("bookDaoFactory");
        System.out.println(bookDao2);
    }
}

3. 实例工厂

在做这个之前复习一下静态方法的构造方式和非静态方法的构造方式:

建立三个类:

StaticClass.java

java 复制代码
package com.example.demo231116;

public class StaticClass {
    public static void staticMethodClass(){
        System.out.println("静态方法构造");
    }
}

UnStaticClass.java

java 复制代码
package com.example.demo231116;

public class UnStaticClass {
    public void unStaticClassMethod(){
        System.out.println("非静态方法");
    }
}

主方法test.java

java 复制代码
package com.example.demo231116;

public class test {

    public static void main(String[] args) {
//        静态方法可以直接调用
        StaticClass.staticMethodClass();

//        非静态方法是不能够直接调用的,需要先创建对象,才能够进行调用
//        UnStaticClass.unStaticMethodClass();
        UnStaticClass mc = new UnStaticClass();
        mc.unStaticClassMethod();

    }
}

这里要注意的是,静态方法能够在不创建类的情况下直接调用,但非静态方法必须要在创建类之后再进行调用,这里bean也是同理,对于实例工厂,其方法不是静态的,我们要先创建工厂,才能够调用工厂下的方法,举例子如下:

工厂类中包含了一个实例方法,非静态

java 复制代码
public class BookDaoFactory {
    public static BookDao getBookDao(){
        System.out.println("Factory method....");
        return new BookDaoImpl();
    }
    
    public BookDao getBookDaoUnstatic(){
        System.out.println("实例工厂方法...");
        return new BookDaoImpl();
    }
}

在Spring config文件中如下写:

xml 复制代码
<bean id="bookDaoFactory2" class="com.example.demo231116.factory.BookDaoFactory" />
<bean id="bd" factory-method="getBookDaoUnstatic" factory-bean="bookDaoFactory2"  />

也就是先创建出这个工厂类,指定id和具体的工厂类class

然后再指定其中具体的工厂方法factory-method,指明这个工厂方法所属的父类factory-bean

调用就直接调用这个bd就可以了

java 复制代码
BookDao bookDao3 = (BookDao) ctx.getBean("bd");
System.out.println(bookDao3);

##########################################

实际上,上面的方法有个没有意义的地方在于,config文件中id为bookDaoFactory2的bean没什么意义,只是为了配合id为bd的bean

为了简化这种两步的配置,有一种最实用的方法,也就是4

4. FactoryBean(实用)

直接写一个工厂方法,implement 一下FactoryBean,发现这个FactoryBean是一个泛型方法,指定其中的类型,实现其方法:

其中第一个getObject()返回具体地对象

第二个返回对象的类型

java 复制代码
package com.example.demo231116.factory;

import com.example.demo231116.dao.BookDao;
import com.example.demo231116.dao.impl.BookDaoImpl;
import org.springframework.beans.factory.FactoryBean;

public class BookDaoFactoryBean implements FactoryBean<BookDao>{
    @Override
    public BookDao getObject() throws Exception {
        return new BookDaoImpl();
    }

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

如此,我们在Spring config中只需要配置一个bean:

xml 复制代码
<bean id="bookDaoFactoryMethod" class="com.example.demo231116.factory.BookDaoFactoryBean" />

在实际过程中直接调用:

java 复制代码
BookDao bookDao4 = (BookDao) ctx.getBean("bookDaoFactoryMethod");
System.out.println(bookDao4);

如果我们写两行使用bookDaoFactoryMethod创建BookDao的方法,再打印两个创建的对象,会发现它们是相同的对象,那么如何创建不同的对象呢?

只需要再补充BookDaoFactoryBean中的这个方法:

java 复制代码
@Override
    public boolean isSingleton() {
        return true;
    }

当这个方法设置为True时,就是用单例模式创建的对象,如果这个方法返回为False,就不会使用单例模式,每一次构造都会创建出新的对象

相关推荐
苍何1 分钟前
终于,我把 Openclaw 加 Seed2.0 Skills 做 AI 漫剧搞定了
后端
Derek_Smart8 分钟前
从一次 OOM 事故说起:打造生产级的 JVM 健康检查组件
java·jvm·spring boot
苍何16 分钟前
阿里出手,最强Coding Plan出炉,OpenClaw可以痛快玩了
后端
风象南36 分钟前
Claude Code这个隐藏技能,让我告别PPT焦虑
人工智能·后端
神奇小汤圆1 小时前
为什么 Spring 强烈推荐你用 singleton
后端
NE_STOP1 小时前
MyBatis-mybatis入门与增删改查
java
Java编程爱好者1 小时前
面试必问:Semaphore 凭什么靠 AQS + CAS 实现限流?
后端
Java编程爱好者2 小时前
十万个why:加了 LIMIT 1,为什么查询反而变慢了?
后端
JavaTalks2 小时前
高并发保护实战:限流、熔断、降级如何配合落地
后端·架构·设计