文章目录
【Java设计模式】多例模式
一、概述
Java中的多例模式确保一个类只有唯一命名的实例,并提供对它们的全局访问点。每个命名实例都通过一个唯一的键进行访问,使其成为Java设计模式的重要组成部分。
二、详细解释及实际示例
- 实际示例:
- 多例模式的一个现实世界示例是大型办公室中的打印机管理系统。在这种情况下,办公室有几台打印机,每台打印机为不同的部门服务。系统不是在每次有打印请求时创建一个新的打印机对象,而是使用多例模式来确保每个部门恰好有一个打印机实例。当来自特定部门的打印请求到来时,系统检查打印机实例的注册表,并检索该部门的现有打印机。如果该部门不存在打印机,则创建一个,注册它,然后返回它。这确保了打印机资源的有效管理,并防止为同一部门不必要地创建多个打印机实例。
- 通俗解释:
- 多例模式是单例模式的扩展,提供了一种拥有唯一命名实例的映射的方式,而不是单个实例。这使其成为有效管理命名实例的有价值的Java设计模式。
- 维基百科解释:
- 在软件工程中,多例模式是对单例模式的一种泛化。单例模式只允许创建一个类的实例,而多例模式允许受控地创建多个实例,并通过使用映射来管理它们。
三、Java中多例模式的编程示例
在本教程中,我们将探讨如何在Java中实现多例模式,涵盖其结构、好处,并提供代码示例。通过遵循这些实现提示,您将能够有效地使用这种Java设计模式。
纳兹古尔,也被称为戒灵或九骑手,是索伦最可怕的仆人。根据定义,他们总是有九个。
Nazgul
是多例类。
public enum NazgulName {
KHAMUL, MURAZOR, DWAR, JI_INDUR, AKHORAHIL, HOARMURATH, ADUNAPHEL, REN, UVATHA
}
public final class Nazgul {
private static final Map<NazgulName, Nazgul> nazguls;
@Getter
private final NazgulName name;
static {
nazguls = new ConcurrentHashMap<>();
nazguls.put(NazgulName.KHAMUL, new Nazgul(NazgulName.KHAMUL));
nazguls.put(NazgulName.MURAZOR, new Nazgul(NazgulName.MURAZOR));
nazguls.put(NazgulName.DWAR, new Nazgul(NazgulName.DWAR));
nazguls.put(NazgulName.JI_INDUR, new Nazgul(NazgulName.JI_INDUR));
nazguls.put(NazgulName.AKHORAHIL, new Nazgul(NazgulName.AKHORAHIL));
nazguls.put(NazgulName.HOARMURATH, new Nazgul(NazgulName.HOARMURATH));
nazguls.put(NazgulName.ADUNAPHEL, new Nazgul(NazgulName.ADUNAPHEL));
nazguls.put(NazgulName.REN, new Nazgul(NazgulName.REN));
nazguls.put(NazgulName.UVATHA, new Nazgul(NazgulName.UVATHA));
}
private Nazgul(NazgulName name) {
this.name = name;
}
public static Nazgul getInstance(NazgulName name) {
return nazguls.get(name);
}
}
下面是我们访问Nazgul
实例的方式。
public static void main(String[] args) {
// 急切初始化的多例
LOGGER.info("打印出急切初始化的多例内容");
LOGGER.info("KHAMUL={}", Nazgul.getInstance(NazgulName.KHAMUL));
LOGGER.info("MURAZOR={}", Nazgul.getInstance(NazgulName.MURAZOR));
LOGGER.info("DWAR={}", Nazgul.getInstance(NazgulName.DWAR));
LOGGER.info("JI_INDUR={}", Nazgul.getInstance(NazgulName.JI_INDUR));
LOGGER.info("AKHORAHIL={}", Nazgul.getInstance(NazgulName.AKHORAHIL));
LOGGER.info("HOARMURATH={}", Nazgul.getInstance(NazgulName.HOARMURATH));
LOGGER.info("ADUNAPHEL={}", Nazgul.getInstance(NazgulName.ADUNAPHEL));
LOGGER.info("REN={}", Nazgul.getInstance(NazgulName.REN));
LOGGER.info("UVATHA={}", Nazgul.getInstance(NazgulName.UVATHA));
// 基于枚举的多例
LOGGER.info("打印出基于枚举的多例内容");
LOGGER.info("KHAMUL={}", NazgulEnum.KHAMUL);
LOGGER.info("MURAZOR={}", NazgulEnum.MURAZOR);
LOGGER.info("DWAR={}", NazgulEnum.DWAR);
LOGGER.info("JI_INDUR={}", NazgulEnum.JI_INDUR);
LOGGER.info("AKHORAHIL={}", NazgulEnum.AKHORAHIL);
LOGGER.info("HOARMURATH={}", NazgulEnum.HOARMURATH);
LOGGER.info("ADUNAPHEL={}", NazgulEnum.ADUNAPHEL);
LOGGER.info("REN={}", NazgulEnum.REN);
LOGGER.info("UVATHA={}", NazgulEnum.UVATHA);
}
程序输出:
15:16:10.597 [main] INFO com.iluwatar.multiton.App -- 打印出急切初始化的多例内容
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- KHAMUL=com.iluwatar.multiton.Nazgul@4141d797
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- MURAZOR=com.iluwatar.multiton.Nazgul@38cccef
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- DWAR=com.iluwatar.multiton.Nazgul@5679c6c6
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- JI_INDUR=com.iluwatar.multiton.Nazgul@27ddd392
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- AKHORAHIL=com.iluwatar.multiton.Nazgul@19e1023e
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- HOARMURATH=com.iluwatar.multiton.Nazgul@7cef4e59
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- ADUNAPHEL=com.iluwatar.multiton.Nazgul@64b8f8f4
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- REN=com.iluwatar.multiton.Nazgul@2db0f6b2
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- UVATHA=com.iluwatar.multiton.Nazgul@3cd1f1c8
15:16:10.600 [main] INFO com.iluwatar.multiton.App -- 打印出基于枚举的多例内容
15:16:10.601 [main] INFO com.iluwatar.multiton.App -- KHAMUL=KHAMUL
15:16:10.601 [main] INFO com.iluwatar.multiton.App -- MURAZOR=MURAZOR
15:16:10.601 [main] INFO com.iluwatar.multiton.App -- DWAR=DWAR
15:16:10.601 [main] INFO com.iluwatar.multiton.App -- JI_INDUR=JI_INDUR
15:16:10.601 [main] INFO com.iluwatar.multiton.App -- AKHORAHIL=AKHORAHIL
15:16:10.601 [main] INFO com.iluwatar.multiton.App -- HOARMURATH=HOARMURATH
15:16:10.601 [main] INFO com.iluwatar.multiton.App -- ADUNAPHEL=ADUNAPHEL
15:16:10.601 [main] INFO com.iluwatar.multiton.App -- REN=REN
15:16:10.601 [main] INFO com.iluwatar.multiton.App -- UVATHA=UVATHA
四、何时在Java中使用多例模式
在Java中使用多例模式的用例:
- 一个类必须有命名实例,但每个唯一键只有一个实例。
- 需要全局访问这些实例,而不需要全局变量。
- 您想要按键管理共享资源。
五、多例模式在Java中的实际应用
- 管理不同上下文中的数据库连接。
- 应用程序中不同环境的配置设置。
六、多例模式的好处和权衡
好处:
- 确保根据键对实例进行受控访问。
- 通过在模式内封装实例管理,减少全局状态的使用。
权衡:
- 如果管理不当,由于多个实例可能会增加内存使用。
- 如果在实现时没有考虑线程安全,可能会出现并发问题。