我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Java工厂模式怎么动态选择对象

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Java工厂模式怎么动态选择对象

本文小编为大家详细介绍“Java工厂模式怎么动态选择对象”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java工厂模式怎么动态选择对象”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

小菜鸟的问题

直接上例子说明,设计一个日志记录的功能,但是支持记录到不同的地方,例如:

  • 内存中

  • 磁盘上的文件

  • 数据库

  • 百度网盘等远程存储服务

面对这么一个需求,你会怎么做呢?我们先来看看小菜鸟的做法吧。

小菜鸟创建了一个Logger

class Logger {    public void log(String message, String loggerMedium) {}}

小菜鸟想都不想,直接一通if else

class Logger {    public void log(String message, String loggerMedium) {        if (loggerMedium.equals("MEMORY")) {            logInMemory(message);        } else if (loggerMedium.equals("FILE")) {            logOnFile(message);        } else if (loggerMedium.equals("DB")) {            logToDB(message);        } else if (loggerMedium.equals("REMOTE_SERVICE")) {            logToRemote(message);        }    }    private void logInMemory(String message) {        // Implementation    }    private void logOnFile(String message) {        // Implementation    }    private void logToDB(String message) {        // Implementation    }    private void logToRemote(String message) {        // Implementation    }}

现在突然说要增加一种存储介质FLASH_DRIVE,就要改了这个类?不拍改错吗?也不符合“开闭原则”,而且随着存储介质变多,类也会变的很大,小菜鸟懵逼了,不知道怎么办?

有没有更好的方法呢

这时候小菜鸟去找你帮忙,你一顿操作,改成了下面这样:

class InMemoryLog {    public void logToMemory(String message) {        // Implementation    }}class FileLog {    public void logToFile(String message) {        //Implementation    }}class DBLog {    public void logToDB(String message) {        // Implementation    }}class RemoteServiceLog {    public void logToService(String message) {        // Implementation    }}class Logger {    private InMemoryLog mLog;    private FileLog fLog;    private DBLog dbLog;    private RemoteServiceLog sLog;        public Logger() {        mLog = new InMemoryLog();        fLog = new FileLog();        dbLog = new DBLog();        sLog = new RemoteServiceLog();    }    public void log(String message, String loggerMedium) {        if (loggerMedium.equals("MEMORY")) {            mLog.logToMemory(message);        } else if (loggerMedium.equals("FILE")) {            fLog.logToFile(message);        } else if (loggerMedium.equals("DB")) {            dbLog.logToDB(message);        } else if (loggerMedium.equals("REMOTE_SERVICE")) {            sLog.logToService(message);        }    }}

在这个实现中,你已经将单独的代码分离到它们对应的文件中,但是Logger类与存储介质的具体实现紧密耦合,如FileLogDBLog等。随着存储介质的增加,类中将引入更多的实例Logger

还有什么更好的办法吗

你想了想,上面的实现都是直接写具体的实现类,是面向实现编程,更合理的做法是面向接口编程,接口意味着协议,契约,是一种更加稳定的方式。

定义一个日志操作的接口

public interface LoggingOperation {    void log(String message);}

实现这个接口

class InMemoryLog implements LoggingOperation {    public void log(String message) {        // Implementation    }}class FileLog implements LoggingOperation {    public void log(String message) {        //Implementation    }}class DBLog implements LoggingOperation {    public void log(String message) {        // Implementation    }}class RemoteServiceLog implements LoggingOperation {    public void log(String message) {        // Implementation    }}

你定义了一个类,据传递的参数,在运行时动态选择具体实现,这就是所谓的工厂类,不过是基础版。

class LoggerFactory {    public static LoggingOperation getInstance(String loggerMedium) {        LoggingOperation op = null;        switch (loggerMedium) {            case "MEMORY":                op = new InMemoryLog();                break;            case "FILE":                op = new FileLog();                break;            case "DB":                op = new DBLog();                break;            case "REMOTE_SERVICE":                op = new RemoteServiceLog();                break;        }        return op;    }}

现在你的 Logger类的实现就是下面这个样子了。

class Logger {    public void log(String message, String loggerMedium) {        LoggingOperation instance = LoggerFactory.getInstance(loggerMedium);        instance.log(message);    }}

这里的代码变得非常统一,创建实际存储实例的责任已经转移到LoggerFactory,各个存储类只实现它们如何将消息记录到它们的特定介质,最后该类Logger只关心通过LoggerFactory将实际的日志记录委托给具体的实现。这样,代码就很松耦合了。你想要添加一个新的存储介质,例如FLASH_DRIVE,只需创建一个实现LoggingOperation接口的新类并将其注册到LoggerFactory中就好了。这就是工厂模式可以帮助您动态选择实现的方式。

还能做得更好吗

你已经完成了一个松耦合的设计,但是想象一下假如有数百个存储介质的场景,所以我们最终会在工厂类LoggerFactory中的switch case部分case数百个。这看起来还是很糟糕,如果管理不当,它有可能成为技术债务,这该怎么办呢?

摆脱不断增长的if else或者 switch case的一种方法是维护类中所有实现类的列表,LoggerFactory代码如下所示:

class LoggerFactory {    private static final List<LoggingOperation> instances = new ArrayList<>();    static {        instances.addAll(Arrays.asList(                new InMemoryLog(),                new FileLog(),                new DBLog(),                new RemoteServiceLog()        ));    }    public static LoggingOperation getInstance(ApplicationContext context, String loggerMedium) {        for(LoggingOperation op : instances) {            // 比如判断StrUtil.equals(loggerMedium, op.getType()) op本身添加一个type        }        return null;    }}

但是请注意,还不够,在所有上述实现中,无论if else、switch case 还是上面的做法,都是让存储实现与LoggerFactory紧密耦合的。你添加一种实现,就要修改LoggerFactory,有什么更好的做法吗?

逆向思维一下,我们是不是让具体的实现主动注册上来呢?通过这种方式,工厂不需要知道系统中有哪些实例可用,而是实例本身会注册并且如果它们在系统中可用,工厂就会为它们提供服务。具体代码如下:

class LoggerFactory {    private static final Map<String, LoggingOperation> instances = new HashMap<>();    public static void register(String loggerMedium, LoggingOperation instance) {        if (loggerMedium != null && instance != null) {            instances.put(loggerMedium, instance);        }    }    public static LoggingOperation getInstance(String loggerMedium) {        if (instances.containsKey(loggerMedium)) {            return instances.get(loggerMedium);        }        return null;    }}

在这里,LoggerFactory提供了一个register注册的方法,具体的存储实现可以调用该方法注册上来,保存在工厂的instancesmap对象中。

我们来看看具体的存储实现注册的代码如下:

class RemoteServiceLog implements LoggingOperation {    static {        LoggerFactory.register("REMOTE", new RemoteServiceLog());    }    public void log(String message) {        // Implementation    }}

由于注册应该只发生一次,所以它发生在static类加载器加载存储类时的块中。

但是又有一个问题,默认情况下JVM不加载类RemoteServiceLog,除非它由应用程序在外部实例化或调用。因此,尽管存储类有注册的代码,但实际上注册并不会发生,因为没有被JVM加载,不会调用static代码块中的代码, 你又犯难了。

你灵机一动,LoggerFactory是获取存储实例的入口点,能否在这个类上做点文章,就写下了下面的代码:

class LoggerFactory {    private static final Map<String, LoggingOperation> instances = new HashMap<>();    static {        try {            loadClasses(LoggerFactory.class.getClassLoader(), "com.alvin.storage.impl");        } catch (Exception e) {            // log or throw exception.        }    }    public static void register(String loggerMedium, LoggingOperation instance) {        if (loggerMedium != null && instance != null) {            instances.put(loggerMedium, instance);        }    }    public static LoggingOperation getInstance(String loggerMedium) {        if (instances.containsKey(loggerMedium)) {            return instances.get(loggerMedium);        }        return null;    }    private static void loadClasses(ClassLoader cl, String packagePath) throws Exception {        String dottedPackage = packagePath.replaceAll("[/]", ".");        URL upackage = cl.getResource(packagePath);        URLConnection conn = upackage.openConnection();        String rr = IOUtils.toString(conn.getInputStream(), "UTF-8");        if (rr != null) {            String[] paths = rr.split("\n");            for (String p : paths) {                if (p.endsWith(".class")) {                    Class.forName(dottedPackage + "." + p.substring(0, p.lastIndexOf('.')));                }            }        }    }}

在上面的实现中,你使用了一个名为loadClasses的方法,该方法扫描提供的包名称com.alvin.storage.impl并将驻留在该目录中的所有类加载到类加载器。以这种方式,当类加载时,它们的static块被初始化并且它们将自己注册到LoggerFactory中。

如何在 SpringBoot 中实现此技术

你突然发现你的应用是springboot应用,突然想到有更方便的解决方案。

因为你的存储实现类都被标记上注解@Component,这样 Spring 会在应用程序启动时自动加载类,它们会自行注册,在这种情况下你不需要使用loadClasses功能,Spring 会负责加载类。具体的代码实现如下:

class LoggerFactory {    private static final Map<String, Class<? extends LoggingOperation>> instances = new HashMap<>();    public static void register(String loggerMedium, Class<? extends LoggingOperation> instance) {        if (loggerMedium != null && instance != null) {            instances.put(loggerMedium, instance);        }    }    public static LoggingOperation getInstance(ApplicationContext context, String loggerMedium) {        if (instances.containsKey(loggerMedium)) {            return context.getBean(instances.get(loggerMedium));        }        return null;    }}

getInstance需要传入ApplicationContext对象,这样就可以根据类型获取具体的实现了。

修改所有存储实现类,如下所示:

import org.springframework.stereotype.Component;@Componentclass RemoteServiceLog implements LoggingOperation {    static {        LoggerFactory.register("REMOTE", RemoteServiceLog.class);    }    public void log(String message) {        // Implementation    }}

读到这里,这篇“Java工厂模式怎么动态选择对象”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Java工厂模式怎么动态选择对象

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Java工厂模式怎么动态选择对象

本文小编为大家详细介绍“Java工厂模式怎么动态选择对象”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java工厂模式怎么动态选择对象”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。小菜鸟的问题直接上例子说明,设
2023-07-05

Java工厂模式用法之如何动态选择对象详解

工厂设计模式可能是最常用的设计模式之一,我想大家在自己的项目中都用到过。本文不仅仅是关于工厂模式的基本知识,更是讨论如何在运行时动态选择不同的方法进行执行,你们可以看看是不是和你们项目中用的一样
2023-03-10

Java简单工厂,工厂方法,抽象工厂模式怎么实现

这篇文章主要讲解了“Java简单工厂,工厂方法,抽象工厂模式怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java简单工厂,工厂方法,抽象工厂模式怎么实现”吧!1、简单工厂模式1.定
2023-06-29

java抽象工厂模式怎么实现

本篇内容介绍了“java抽象工厂模式怎么实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!抽象工厂只实现了一个 createHuman 的方
2023-06-19

java工厂模式创建对象的方法是什么

Java工厂模式创建对象的方法是通过工厂类来创建对象。工厂类根据客户端的需求,创建一种或多种不同类型的对象。这种方法将对象的实例化过程封装到工厂类中,使得客户端只需与工厂类交互,而不需要直接创建对象。工厂模式可以根据需求返回不同类型的对象,
2023-10-20

怎么理解Java设计模式的抽象工厂模式

这篇文章主要讲解了“怎么理解Java设计模式的抽象工厂模式”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解Java设计模式的抽象工厂模式”吧!一、什么是抽象工厂模式为创建一组相关或相互
2023-06-25

java工厂模式怎么写

这篇文章主要介绍“java工厂模式怎么写”,在日常操作中,相信很多人在java工厂模式怎么写问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”java工厂模式怎么写”的疑惑有所帮助!接下来,请跟着小编一起来学习吧
2023-06-19

怎么用Java代码实现抽象工厂模式

这篇文章主要介绍“怎么用Java代码实现抽象工厂模式”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么用Java代码实现抽象工厂模式”文章能帮助大家解决问题。解决工厂方法模式的问题:一个具体工厂只能
2023-06-29

JAVA抽象工厂模式是什么及怎么实现

这篇文章主要讲解了“JAVA抽象工厂模式是什么及怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“JAVA抽象工厂模式是什么及怎么实现”吧!什么是抽象工厂模式抽象工厂(AbstractF
2023-06-30

php抽象工厂模式怎么应用

抽象工厂模式是一种创建对象的设计模式,它提供了一种封装一组具有相似功能的工厂的方式。在PHP中,可以通过以下步骤应用抽象工厂模式:1. 定义抽象工厂接口:创建一个抽象工厂接口,用于声明创建产品的方法。```phpinterface Abst
2023-08-30

Python怎么实现抽象工厂模式

这篇文章主要介绍“Python怎么实现抽象工厂模式”,在日常操作中,相信很多人在Python怎么实现抽象工厂模式问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python怎么实现抽象工厂模式”的疑惑有所帮助!
2023-06-19

java的工厂模式怎么实现

在Java中,工厂模式可以通过两种方式来实现:简单工厂模式和抽象工厂模式。简单工厂模式:简单工厂模式的核心是定义一个工厂类,该类负责创建其他类的实例。具体步骤如下:创建一个抽象产品类,该类定义了产品的共同属性和方法。创建具体产品类,继
java的工厂模式怎么实现
2024-02-29

.Net设计模式之抽象工厂模式怎么实现

这篇文章主要讲解了“.Net设计模式之抽象工厂模式怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“.Net设计模式之抽象工厂模式怎么实现”吧!一、动机(Motivation)在软件系统
2023-06-30

Java设计模式的工厂模式怎么实现

本篇内容介绍了“Java设计模式的工厂模式怎么实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!工厂模式在Java应用程序中对象无处不在,这
2023-06-29

java简单工厂模式怎么实现

在Java中,简单工厂模式可以通过以下步骤实现:1. 首先创建一个接口,表示产品的抽象类,其中定义了产品的一些共同的方法。```javapublic interface Product {void use();}```2. 实现产品接口的具
2023-08-18

Java工厂方法模式怎么实现

这篇文章主要讲解了“Java工厂方法模式怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java工厂方法模式怎么实现”吧!一、什么是工厂方法模式?工厂方法模式:也叫虚拟构造器模式或者多
2023-06-29

Java怎么实现工厂方法设计模式

这篇文章主要介绍“Java怎么实现工厂方法设计模式”,在日常操作中,相信很多人在Java怎么实现工厂方法设计模式问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java怎么实现工厂方法设计模式”的疑惑有所帮助!
2023-06-04

C#中对象状态模式怎么实现

这篇文章主要介绍了C#中对象状态模式怎么实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C#中对象状态模式怎么实现文章都会有所收获,下面我们一起来看看吧。定义枚举表示教程进度首先用一个枚举,表示教程进行的不同
2023-06-30

怎么用Java代码实现工厂方法模式

今天小编给大家分享一下怎么用Java代码实现工厂方法模式的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1. 工厂方法模式介绍
2023-06-29

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录