Spring结合工厂模式

学习设计模式,不要进入一个误区生搬硬套,它是一种编程思想,结合实际使用,往往设计模式是混合使用的

工厂模式

核心本质: 使用工厂统一管理对象的创建,将调用者跟实现类解耦

我这里使用Spring容器的支持,实现起来将更加简单,只需要实现产品接口,就能产品就能自动注册到工厂,让工厂统一管理。以后有新的产品,不修改原来的代码,新产品实现产品接口,专注新产品的业务功能开发即可。

java 复制代码
package com.dj.factory;

/**
 * User: ldj
 * Date: 2024/1/28
 * Time: 13:38
 * Description: 产品类
 */
public interface Animal {

    void description();
}


@Component
public class Cat implements Animal {

    @Override
    public void description() {
        System.out.println("我是一只可爱的喵星人");
    }
}


@Component
public class Dog implements Animal {

    @Override
    public void description() {
        System.out.println("我是一只爱拆家的二哈");
    }
}
java 复制代码
package com.dj.factory;

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
import org.springframework.util.CollectionUtils;

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * User: ldj
 * Date: 2024/1/28
 * Time: 13:11
 * Description: 工厂类
 */
@Component
public class AnimalFactory implements ApplicationContextAware {

    private static final Map<String, Animal> animalMap = new ConcurrentHashMap<>();

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        register(applicationContext);
    }

    public static Animal getInstance(String beanName) {
        return animalMap.get(beanName);
    }

    public static void register(ApplicationContext applicationContext) {
        Map<String, Animal> beanMap = applicationContext.getBeansOfType(Animal.class);
        if (!CollectionUtils.isEmpty(beanMap)) {
            animalMap.putAll(beanMap);
        }
    }
}

其实注册这个动作还可以在产品类实现接口 InitializingBean 在重写方法进行注入类似这样:

还可以使用 @PostConstruct 自己写init方法,在init方法方法里调用工厂提供的注入方法

那么问题来了,有Spring容器管理的Animal的实现类我们还要必要自己写工厂类吗?于是继续优化代码:

java 复制代码
package com.dj.factory;

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

/**
 * User: ldj
 * Date: 2024/1/28
 * Time: 13:11
 * Description: 工厂类
 */
@Component
public class AnimalFactory implements ApplicationContextAware {

    private static ApplicationContext applicationContext;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        AnimalFactory.applicationContext = applicationContext;
    }

    public static Animal getInstance(String beanName) {
        return (Animal) applicationContext.getBean(beanName);
    }

}
相关推荐
wand codemonkey7 分钟前
SpringbootWeb【入门】+MySQL【安装】+【DataDrip安装 】+【连接MySQL】
java·mysql·mybatis
Mahir088 小时前
Spring 循环依赖深度解密:从问题本质到三级缓存源码级解析
java·后端·spring·缓存·面试·循环依赖·三级缓存
RyFit9 小时前
SpringAI 常见问题及解决方案大全
java·ai
石山代码9 小时前
C++ 内存分区 堆区
java·开发语言·c++
绝知此事10 小时前
【算法突围 01】线性结构与哈希表:后端开发的收纳术
java·数据结构·算法·面试·jdk·散列表
无风听海10 小时前
C# 隐式转换深度解析
java·开发语言·c#
一只大袋鼠10 小时前
Git 进阶(二):分支管理、暂存栈、远程仓库与多人协作
java·开发语言·git
德思特11 小时前
从 Dify 配置页理解 RAG 的重要参数
java·人工智能·llm·dify·rag
YOU OU11 小时前
Spring IoC&DI
java·数据库·spring
один but you12 小时前
从可变参数到 emplace:现代 C++ 性能优化的核心组合
java·开发语言