chapter03_Bean的实例化与策略模式

一、Bean的实例化

  • Spring中的获取的Bean,可能是原始对象,也可能是代理对象(JDK或CGLIB)
  • 所以我们应该至少提供两种实例化的方式,直接new或者是CGLIB

二、策略模式

由于实例化的方式很多,我们这里使用策略模式,便于以后不断地扩展

先扩展一下BeanFactory接口,获取bean的时候,可以传入参数,这样就可以使用带参数的构造器

java 复制代码
public interface BeanFactory {

    Object getBean(String name);

    Object getBean(String name, Object... args);
}

定义实例化策略接口

java 复制代码
public interface InstantiationStrategy {
    Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args);
}

改造doCreateBean方法,灵活切换对象实例化的方式

  • 此处使用最简单的逻辑,直接根据传入的参数数量,匹配对应的构造器
java 复制代码
/**
 * 负责创建Bean
 *
 * @Author 孤风雪影
 * @Email gitee.com/efairy520
 * @Date 2025/1/1 1:01
 * @Version 1.0
 */
public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory {

    private final InstantiationStrategy instantiationStrategy = new CglibSubclassingInstantiationStrategy();

    /**
     * 实现父类未实现的createBean方法
     * @param beanName
     * @param beanDefinition
     * @return
     */
    @Override
    protected Object createBean(String beanName, BeanDefinition beanDefinition, Object[] args) {
        Object bean = doCreateBean(beanName, beanDefinition, args);
        addSingleton(beanName, bean);
        return bean;
    }

    protected Object doCreateBean(String beanName, BeanDefinition beanDefinition, Object[] args) {
        Constructor constructorToUse = null;
        Class<?> beanClass = beanDefinition.getBeanClass();
        Constructor<?>[] declaredConstructors = beanClass.getDeclaredConstructors();
        for (Constructor<?> ctor : declaredConstructors) {
            //简易逻辑,只比较数量
            if (null != args && ctor.getParameterTypes().length == args.length) {
                constructorToUse = ctor;
                break;
            }
        }
        return instantiationStrategy.instantiate(beanDefinition, beanName, constructorToUse, args);
    }
}

三、简单实例化策略

使用传入的构造器和传入的参数表,直接new对象

java 复制代码
/**
 * @Author 孤风雪影
 * @Email gitee.com/efairy520
 * @Date 2025/1/2 0:01
 * @Version 1.0
 */
public class SimpleInstantiationStrategy implements InstantiationStrategy {
    @Override
    public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args) {
        Class clazz = beanDefinition.getBeanClass();
        try {
            if (ctor != null) {
                return clazz.getDeclaredConstructor(ctor.getParameterTypes()).newInstance(args);
            } else {
                return clazz.getDeclaredConstructor().newInstance();
            }
        } catch (Exception e) {
            throw new RuntimeException("Failed to instantiate" + e);
        }
    }
}

四、CGLIB实例化策略

使用传入的构造器和传入的参数表,创建CGLIB代理对象

java 复制代码
/**
 * @Author 孤风雪影
 * @Email gitee.com/efairy520
 * @Date 2025/1/2 0:12
 * @Version 1.0
 */
public class CglibSubclassingInstantiationStrategy implements InstantiationStrategy{
    @Override
    public Object instantiate(BeanDefinition beanDefinition, String beanName, Constructor ctor, Object[] args) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(beanDefinition.getBeanClass());
        enhancer.setCallback(new NoOp() {
            @Override
            public int hashCode() {
                return super.hashCode();
            }
        });
        if (null == ctor) return enhancer.create();
        return enhancer.create(ctor.getParameterTypes(), args);
    }
}

五、测试

给测试的Bean,新建空参构造器和有参构造器

java 复制代码
package cn.shopifymall.springframework.test.bean;

/**
 * @Author 孤风雪影
 * @Email gitee.com/efairy520
 * @Date 2025/2/6 23:15
 * @Version 1.0
 */
public class UserService {

    private String name;

    public UserService() {
        this.name = "undefined";
    }

    public UserService(String name) {
        this.name = name;
    }

    public void queryUserInfo() {
        System.out.println("查询用户信息:" + name);
    }

    @Override
    public String toString() {
        final StringBuilder sb = new StringBuilder("");
        sb.append("").append(name);
        return sb.toString();
    }
}

测试代码

java 复制代码
public class ApiTest {

    @Test
    public void test_BeanFactory(){
        // 1.初始化 BeanFactory
        DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory();

        // 2. 注入bean
        BeanDefinition beanDefinition = new BeanDefinition(UserService.class);
        beanFactory.registerBeanDefinition("userService", beanDefinition);

        // 3.获取bean
        UserService userService = (UserService) beanFactory.getBean("userService", "abc");
        userService.queryUserInfo();
    }
}

测试以下四种场景

  • 简单策略空参
  • 简单策略带参
  • CGLIB策略空参
  • CGLIB策略带参
相关推荐
朦胧之9 小时前
AI 编程-老项目改造篇
java·前端·后端
程序猿大帅14 小时前
别再只当调包侠了:用 Spring AI 落地 Function Calling,我被大模型硬生生砸出了三个大坑
java
程序员晓琪15 小时前
约定大于配置:基于 Java 包名自动生成 API 版本路由的最佳实践
java·spring boot·后端
Flittly15 小时前
【AgentScope Java新手村系列】(11)中断与恢复
java·spring boot·spring
众少成多积小致巨15 小时前
JNI (Java Native Interface) 技术手册中文参考指南
android·java·c++
东坡白菜16 小时前
破局全栈:前端开发的Java入门实战记录—JPA(2)
java·后端
SimonKing1 天前
艹,维护AI写的代码,我心态崩了......
java·后端·程序员
用户298698530141 天前
Java Word 文档样式进阶:段落与文本背景色设置完全指南
java·后端
dunky1 天前
Spring 的三级缓存与循环依赖
后端·spring
小bo波2 天前
从"任意文件复制"深挖Java I/O:字符流与字节流的本质抉择
java·nio·io流·后端开发·文件复制