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);
    }

}
相关推荐
雨中飘荡的记忆2 小时前
ElasticJob分布式调度从入门到实战
java·后端
考虑考虑11 小时前
JDK25模块导入声明
java·后端·java ee
_小马快跑_12 小时前
Java 的 8 大基本数据类型:为何是不可或缺的设计?
java
Re_zero14 小时前
线上日志被清空?这段仅10行的 IO 代码里竟然藏着3个毒瘤
java·后端
洋洋技术笔记14 小时前
Spring Boot条件注解详解
java·spring boot
程序员清风1 天前
程序员兼职必看:靠谱软件外包平台挑选指南与避坑清单!
java·后端·面试
皮皮林5511 天前
利用闲置 Mac 从零部署 OpenClaw 教程 !
java
华仔啊2 天前
挖到了 1 个 Java 小特性:var,用完就回不去了
java·后端
SimonKing2 天前
SpringBoot整合秘笈:让Mybatis用上Calcite,实现统一SQL查询
java·后端·程序员
日月云棠2 天前
各版本JDK对比:JDK 25 特性详解
java