【设计模式】享元模式 实现 ( 实现流程 | 抽象享元类 | 具体享元类 | 享元工厂 | 用户调用 | 代码模板 )

简介: 【设计模式】享元模式 实现 ( 实现流程 | 抽象享元类 | 具体享元类 | 享元工厂 | 用户调用 | 代码模板 )

文章目录

I . 享元模式 实现流程

II . 享元模式 抽象享元类

III . 享元模式 具体享元类

IV . 享元模式 享元工厂

V . 享元模式 用户调用



I . 享元模式 实现流程


1 . 定义抽象享元类 : 定义抽象类 , 内部状态 , 外部状态 , 抽象方法 ;



① 抽象类 : 该类是一个 抽象类 , 在 享元工厂类 和 用户调用 中 , 涉及的对象类型就声明为该 抽象类型 , 一般不直接使用具体的实现类 ;


② 定义内部状态 : 内部状态数据是多个 细粒度对象 ( 用户实际使用的一万个对象 ) 共享的信息 , 共享就是在对象池中的维护一个享元对象 ( 共享对象 ) , 当多个 细粒度对象 从 享元工厂类 中获取该享元对象时 , 获取的都是该享元对象 ( 共享对象 ) , 这些细粒度对象 ( 用户使用的对象 ) 使用的内部状态 ( 数据 ) 肯定都是一样的 ; 定义时不用太关心该状态 , 区分内部状态与外部状态即可 ;


③ 定义外部状态 : 该状态很重要 , 该信息作为从对象池中获取对象的依据 ; 即外部状态相等的两个对象 , 说明对象是相同的 ;


④ 定义抽象行为 : 享元类实际的操作 , 此处是抽象方法 , 也是最终用户调用该享元类调用的方法 ;



2 . 定义具体享元类 : 继承抽象享元类 , 实现抽象享元类的抽象方法 ;



① 继承 : 该类继承 抽象的享元类 , 可以定义多个具体享元类 ;


② 使用 : 声明享元类对象时 , 声明抽象享元类类型 , 实际为该对象赋值时 , 需要赋值具体的享元类 ;


③ 对象池对象 : 享元工厂类中的对象池 , 可以存储不同的具体享元类对象 , 但必须都是抽象享元类的子类对象 ;



3 . 定义享元工厂类 :



① 定义对象池 : 对象池一般是 Map 键值对类型 , 使用 抽象享元类对象的 外部状态 数据作为键 , 值就是 享元类对象 ;


② 对象获取机制 : 当用户根据 外部状态 获取享元类对象时 , 先在对象池中查找是否存在该享元对象 , 如果存在直接返回该对象 , 如果不存在 , 那么创建该对象 , 将新创建的对象放入对象池中 , 并返回该享元对象 ;



4 . 用户调用 : 用户使用 享元工厂类 创建 抽象享元类 类型对象 , 并为其赋值一个 具体的享元类 对象 , 调用 抽象的方法 ;




II . 享元模式 抽象享元类


1 . 定义 抽象享元类 AbstractFlyweight :



① 定义内部状态 : private String intrinsicState是内部状态 , 也就是多个 细粒度对象 ( 用户持有的多个对象 ) 共享的信息 , 这部分内容我们不关心 , 为其设置 Getter 和 Setter 方法即可 ;


② 定义外部状态 : protected final String externalState , 外部状态推荐使用 final 修饰 , 这是为了避免其在构造函数外的地方修改该值 , 外部状态值只能在对象创建的时候赋值一次 , 其它任何时候都不允许修改该值 ;


③ 定义抽象行为 : abstract void action() , 这是享元类的具体逻辑 , 在具体享元角色中实现该类 ;



2 . 抽象享元类 代码示例 :


package kim.hsl.design.flyweight;
/**
 * 抽象的享元角色
 */
public abstract class AbstractFlyweight {
    /**
     * 内部状态
     */
    private String intrinsicState;
    /**
     * 外部状态
     * 注意使用 final 修饰只能 ① 马上初始化 ② 在构造函数中初始化
     * 没有其它的初始化途径
     */
    protected final String externalState;
    public AbstractFlyweight(String externalState) {
        this.externalState = externalState;
    }
    /**
     * 抽象的行为
     */
    abstract void action();
    /*
        下面是访问内部状态的 Getter 和 Setter 方法
     */
    public String getIntrinsicState() {
        return intrinsicState;
    }
    public void setIntrinsicState(String intrinsicState) {
        this.intrinsicState = intrinsicState;
    }
    /**
     * 获取外部状态的数据
     * 该数据用于 作为 对比两个对象是否一致 的依据
     * @return
     */
    public String getExternalState() {
        return externalState;
    }
}





III . 享元模式 具体享元类


1 . 定义具体享元类 ConcreteFlyweight :



① 继承 : 该类继承 AbstractFlyweight 抽象类 ;


② 重写构造函数 : 抽象类实现类需要重写有参构造函数 ;


③ 实现抽象方法 : 具体的享元类中需要实现 抽象享元类中的 抽象行为 ;



2 . 具体享元类 代码示例 :


package kim.hsl.design.flyweight;
/**
 * 具体的享元角色
 * 注意要实现 抽象方法 , 覆盖其非默认的构造函数
 */
public class ConcreteFlyweight extends AbstractFlyweight {
    public ConcreteFlyweight(String externalState) {
        super(externalState);
    }
    @Override
    void action() {
        System.out.println("享元对象行为 , 外部状态 : " + externalState);
    }
}



IV . 享元模式 享元工厂


1 . 定义 享元工厂类 :


① 对象池 : 享元工厂类 中维护一个静态的 HashMap<String, AbstractFlyweight> 数据结构 , 这就是对象池在内存中的数据结构 , 键 String 数据就是 享元类的 外部状态 , 用于作为 享元对象 的 身份标识 , 每个享元对象的该 外部状态值 都不一样 ;


② 从对象池中获取对象 : 从 HashMap<String, AbstractFlyweight> OBJECT_POOL 对象池中获取 AbstractFlyweight 抽象享元类 对象时 , 先检查对象池中是否有该数据 , 如果使用 map.get() 获取的 享元对象不为空 , 说明对象池中有该数据 , 直接返回即可 ; 如果 map.get() 获取的 享元对象为空 , 说明对象池中没有该对象 , 需要创建该对象 , 放入对象池 , 然后返回 ; 注意要使用 外部状态传入构造函数创建对象 ;



2 . 享元工厂类 代码示例 :


package kim.hsl.design.flyweight;
import java.util.HashMap;
/**
 * 享元工厂
 * 维护对象池
 */
public class FlyweightFactory {
    /**
     * 对象池
     * 使用对象时, 从对象池中查找
     * 如果找到直接取出
     * 如果没有找到, 创建一个新对象, 放入对象池中, 然后返回该对象
     * 键 : 外部状态 值 : 享元对象
     */
    private static final HashMap<String, AbstractFlyweight> OBJECT_POOL = new HashMap<String, AbstractFlyweight>();
    public static AbstractFlyweight getObject(String externalState){
        //从对象池中获取对象
        ConcreteFlyweight object = (ConcreteFlyweight) OBJECT_POOL.get(externalState);
        //如果对象池中没有该对象
        if(object == null){
            object = new ConcreteFlyweight(externalState);
            OBJECT_POOL.put(externalState, object);
            System.out.println("创建新对象 外部状态 值 : " + externalState + " , 并放入对象池");
        }else{
            System.out.println("获取对象池中已有的对象 , 外部状态 值 : " + externalState);
        }
        return object;
    }
}




V . 享元模式 用户调用


1 . 用户调用 :



① 用户声明对象类型 : 用户声明 AbstractFlyweight 抽象享元类的对象 , 从享元工厂类的对象池中获取对象 ;


② 新建对象 : 如果对象池中没有对象 , 会调用构造函数 , 创建该对象 , 并放入对象池中 , 再返回该对象 ;


② 复用对象 : 如果对象池中有该享元对象 , 那么直接从线程池中获取该对象 , 并返回 ;



2 . 用户调用 代码示例 :


package kim.hsl.design.flyweight;
public class Main {
    /**
     * 注意 : 享元对象使用时, 创建的是抽象的享元类
     * @param args
     */
    public static void main(String[] args) {
        //外部状态值为 "tom" , 对象池中没有该对象, 创建后放入对象池, 然后从对象池中取出使用
        AbstractFlyweight tom = FlyweightFactory.getObject("tom");
        tom.action();
        //外部状态值为 "tom" 的对象之前已经创建过, 直接从对象池中取出使用
        AbstractFlyweight tom2 = FlyweightFactory.getObject("tom");
        tom2.action();
        //外部状态值为 "Jerry" , 对象池中没有该对象, 创建后放入对象池, 然后从对象池中取出使用
        AbstractFlyweight jerry = FlyweightFactory.getObject("Jerry");
        jerry.action();
    }
}



目录
相关文章
|
3月前
|
设计模式 数据库连接 PHP
PHP中的设计模式:提升代码的可维护性与扩展性在软件开发过程中,设计模式是开发者们经常用到的工具之一。它们提供了经过验证的解决方案,可以帮助我们解决常见的软件设计问题。本文将介绍PHP中常用的设计模式,以及如何利用这些模式来提高代码的可维护性和扩展性。我们将从基础的设计模式入手,逐步深入到更复杂的应用场景。通过实际案例分析,读者可以更好地理解如何在PHP开发中应用这些设计模式,从而写出更加高效、灵活和易于维护的代码。
本文探讨了PHP中常用的设计模式及其在实际项目中的应用。内容涵盖设计模式的基本概念、分类和具体使用场景,重点介绍了单例模式、工厂模式和观察者模式等常见模式。通过具体的代码示例,展示了如何在PHP项目中有效利用设计模式来提升代码的可维护性和扩展性。文章还讨论了设计模式的选择原则和注意事项,帮助开发者在不同情境下做出最佳决策。
|
1月前
|
设计模式 前端开发 JavaScript
JavaScript设计模式及其在实战中的应用,涵盖单例、工厂、观察者、装饰器和策略模式
本文深入探讨了JavaScript设计模式及其在实战中的应用,涵盖单例、工厂、观察者、装饰器和策略模式,结合电商网站案例,展示了设计模式如何提升代码的可维护性、扩展性和可读性,强调了其在前端开发中的重要性。
30 2
|
2月前
|
设计模式 算法 数据库连接
PHP中的设计模式:提高代码的可维护性和扩展性
【10月更文挑战第13天】 本文将探讨PHP中常见的设计模式及其在实际项目中的应用。通过对比传统编程方式,我们将展示设计模式如何有效地提高代码的可维护性和扩展性。无论是单例模式确保类的单一实例,还是观察者模式实现对象间的松耦合,每一种设计模式都为开发者提供了解决特定问题的最佳实践。阅读本文后,读者将能更好地理解和应用这些设计模式,从而提升PHP编程的效率和质量。
|
2月前
|
设计模式 SQL 安全
PHP中的设计模式:单例模式的深入探索与实践在PHP开发领域,设计模式是解决常见问题的高效方案集合。它们不是具体的代码,而是一种编码和设计经验的总结。单例模式作为设计模式中的一种,确保了一个类仅有一个实例,并提供一个全局访问点。本文将深入探讨单例模式的基本概念、实现方式及其在PHP中的应用。
单例模式在PHP中的应用广泛,尤其在处理数据库连接、日志记录等场景时,能显著提高资源利用率和执行效率。本文从单例模式的定义出发,详细解释了其在PHP中的不同实现方法,并探讨了使用单例模式的优势与注意事项。通过对示例代码的分析,读者将能够理解如何在PHP项目中有效应用单例模式。
|
3月前
|
设计模式 算法 数据库连接
PHP中的设计模式:提高代码的可维护性与扩展性
设计模式在PHP开发中至关重要,如单例模式确保类仅有一个实例并提供全局访问点,适用于管理数据库连接或日志记录。工厂模式封装对象创建过程,降低系统耦合度;策略模式定义算法系列并使其可互换,便于实现不同算法间的切换。合理选择设计模式需基于需求分析,考虑系统架构,并通过测试驱动开发验证有效性,确保团队协作一致性和代码持续优化。设计模式能显著提升代码质量,解决开发中的设计难题。
40 8
|
3月前
|
设计模式 算法 PHP
PHP中的设计模式:提升代码的灵活性与可维护性
在本文中,我们将深入探讨PHP编程语言中的一种重要概念——设计模式。设计模式是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。它代表了最佳的实践,被有经验的面向对象的软件开发人员所采用。本文将通过具体的实例,展示如何在PHP项目中应用设计模式,以提高代码的灵活性和可维护性。无论你是PHP初学者还是经验丰富的开发者,都能从中获得有价值的见解。
|
3月前
|
设计模式 算法 PHP
PHP中的设计模式:策略模式的深入探索与实践在软件开发的广袤天地中,PHP以其独特的魅力和强大的功能,成为无数开发者手中的得力工具。而在这条充满挑战与机遇的征途上,设计模式犹如一盏明灯,指引着我们穿越代码的迷雾,编写出更加高效、灵活且易于维护的程序。今天,就让我们聚焦于设计模式中的璀璨明珠——策略模式,深入探讨其在PHP中的实现方法及其实际应用价值。
策略模式,这一设计模式的核心在于它为软件设计带来了一种全新的视角和方法。它允许我们在运行时根据不同情况选择最适合的解决方案,从而极大地提高了程序的灵活性和可扩展性。在PHP这门广泛应用的编程语言中,策略模式同样大放异彩,为开发者们提供了丰富的创作空间。本文将从策略模式的基本概念入手,逐步深入到PHP中的实现细节,并通过一个具体的实例来展示其在实际项目中的应用效果。我们还将探讨策略模式的优势以及在实际应用中可能遇到的挑战和解决方案,为PHP开发者提供一份宝贵的参考。
|
3月前
|
设计模式 存储 数据库连接
探索PHP中的设计模式:提高代码的可维护性与扩展性
本文将深入探讨PHP中常用的设计模式,包括单例模式、工厂模式和观察者模式。通过具体的代码示例,展示如何在实际项目中应用这些设计模式,以提高代码的可维护性与扩展性。无论你是PHP初学者还是有一定经验的开发者,都可以通过本文的学习,提升你的编程技巧和项目架构能力。
|
3月前
|
设计模式 算法 搜索推荐
PHP中的设计模式:提高代码可维护性的秘诀
在本文中,我们将探讨PHP设计模式的重要性以及它们如何帮助开发者编写出更加灵活、可维护的代码。我们将介绍几种常见的设计模式,包括单例模式、工厂模式和策略模式,并通过实际示例展示它们在PHP中的应用。最后,我们会讨论何时使用这些设计模式以及在实际项目开发中的最佳实践。
50 1
|
3月前
|
设计模式 Java
Java设计模式-享元模式(12)
Java设计模式-享元模式(12)

热门文章

最新文章