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

}
相关推荐
007php00718 分钟前
Redis高级面试题解析:深入理解Redis的工作原理与优化策略
java·开发语言·redis·nginx·缓存·面试·职场和发展
Yeats_Liao37 分钟前
Spring缓存(二):解决缓存雪崩、击穿、穿透问题
java·spring·缓存
猿究院-赵晨鹤1 小时前
String、StringBuffer 和 StringBuilder 的区别
java·开发语言
葵野寺1 小时前
【RelayMQ】基于 Java 实现轻量级消息队列(九)
java·开发语言·rabbitmq·java-rabbitmq
代码不停1 小时前
MySQL联合查询
java·数据库·mysql
nightunderblackcat1 小时前
新手向:C语言、Java、Python 的选择与未来指南
java·c语言·python
纯真时光1 小时前
Maven高级
java
好多172 小时前
《微服务事务管理》
java·微服务·架构
llp11102 小时前
MQTT Dashboard
java
浪扼飞舟2 小时前
c#基础二(类和对象,构造器调用顺序、访问级别、重写和多态、抽象类和接口)
java·开发语言·c#