草庐IT

享元模式

Whyn 2023-09-21 原文

简介

Use sharing to support large numbers of fine-grained objects efficiently.
使用共享对象可有效地支持大量的细粒度的对象。

享元模式(Flyweight)又称为 轻量级模式,它是一种对象结构型模式。

面向对象技术可以很好地解决一些灵活性或可扩展性问题,但在很多情况下需要在系统中增加类和对象的个数。当对象数量太多时,将导致运行代价过高,带来性能下降等问题。享元模式 正是为解决这一类问题而诞生的。

享元模式 是对象池的一种实现。类似于线程池,线程池可以避免不停的创建和销毁多个对象,消耗性能。享元模式 也是为了减少内存的使用,避免出现大量重复的创建销毁对象的场景。

享元模式 的宗旨是共享细粒度对象,将多个对同一对象的访问集中起来,不必为每个访问者创建一个单独的对象,以此来降低内存的消耗。

享元模式 把一个对象的状态分成内部状态和外部状态,内部状态即是不变的,外部状态是变化的;然后通过共享不变的部分,达到减少对象数量并节约内存的目的。

享元模式 本质:缓存共享对象,降低内存消耗

主要解决

当系统中多处需要同一组信息时,可以把这些信息封装到一个对象中,然后对该对象进行缓存,这样,一个对象就可以提供给多处需要使用的地方,避免大量同一对象的多次创建,消耗大量内存空间。

享元模式 其实就是 工厂模式 的一个改进机制,享元模式 同样要求创建一个或一组对象,并且就是通过工厂方法生成对象的,只不过 享元模式 中为工厂方法增加了缓存这一功能。

优缺点

优点

  • 享元模式 可以极大减少内存中对象的数量,使得相同对象或相似对象在内存中只保存一份,降低内存占用,增强程序的性能;
  • 享元模式 的外部状态相对独立,而且不会影响其内部状态,从而使得享元对象可以在不同的环境中被共享;

缺点

  • 享元模式 使得系统更加复杂,需要分离出内部状态和外部状态,这使得程序的逻辑复杂化;
  • 为了使对象可以共享,享元模式 需要将享元对象的状态外部化,而且外部状态必须具备固化特性,不应该随内部状态改变而改变,否则会导致系统的逻辑混乱;

使用场景

  • 系统中存在大量的相似对象;
  • 细粒度的对象都具备较接近的外部状态,而且内部状态与环境无关,也就是说对象没有特定身份;
  • 需要缓冲池的场景;

模式讲解

首先来看下 享元模式 的通用 UML 类图:

享元模式

从 UML 类图中,我们可以看到,享元模式 主要包含三种角色:

  • 抽象享元角色(Flyweight):享元对象抽象基类或者接口,同时定义出对象的外部状态和内部状态的接口或实现;
  • 具体享元角色(ConcreteFlyweight):实现抽象角色定义的业务。该角色的内部状态处理应该与环境无关,不能出现会有一个操作改变内部状态,同时修改了外部状态;
  • 享元工厂(FlyweightFactory):负责管理享元对象池和创建享元对象;

以下是 享元模式 的通用代码:

class Client {
    public static void main(String[] args) {
        IFlyweight flyweight1 = FlyweightFactory.getFlyweight("aa");
        IFlyweight flyweight2 = FlyweightFactory.getFlyweight("bb");
        flyweight1.operation("a");
        flyweight2.operation("b");
    }

    // 抽象享元角色
    interface IFlyweight {
        void operation(String extrinsicState);
    }

    // 具体享元角色
    static class ConcreteFlyweight implements IFlyweight {
        private String intrinsicState;

        public ConcreteFlyweight(String intrinsicState) {
            this.intrinsicState = intrinsicState;
        }

        @Override
        public void operation(String extrinsicState) {
            System.out.println("Object address: " + System.identityHashCode(this));
            System.out.println("IntrinsicState: " + this.intrinsicState);
            System.out.println("ExtrinsicState: " + extrinsicState);
        }
    }

    // 享元工厂
    static class FlyweightFactory {
        private static Map<String, IFlyweight> pool = new HashMap<>();

        // 因为内部状态具备不变性,因此作为缓存的键
        public static IFlyweight getFlyweight(String intrinsicState) {
            if (!pool.containsKey(intrinsicState)) {
                IFlyweight flyweight = new ConcreteFlyweight(intrinsicState);
                pool.put(intrinsicState, flyweight);
            }
            return pool.get(intrinsicState);
        }
    }
}

举个例子

例子:我们知道,过年回家的时候很麻烦,因为我们需要抢到一张回家的火车票。抢票的时候,我们肯定是要查询下有没有我们需要的票信息,这里我们假设一张火车的信息包含:出发站,目的站,价格,座位类别。现在要求编写一个查询火车票查询伪代码,可以通过出发站,目的站查到相关票的信息。

直接思路:例子要求通过出发站,目的站查询火车票的相关信息,那么我们只需构建出火车票类对象,然后提供一个查询出发站,目的站的接口给到客户进行查询即可;
具体代码如下:

class Client {
    public static void main(String[] args) {
        ITicket ticket = TicketFactory.queryTicket("深圳北", "潮汕");
        ticket.showInfo("硬座");
    }

    interface ITicket {
        void showInfo(String bunk);
    }

    static class TrainTicket implements ITicket {
        private String from;
        private String to;

        public TrainTicket(String from, String to) {
            this.from = from;
            this.to = to;
        }

        @Override
        public void showInfo(String bunk) {
            int price = new Random().nextInt(500);
            System.out.println(String.format("%s->%s:%s价格:%s 元", this.from, this.to, bunk, price));
        }
    }

    static class TicketFactory {
        public static ITicket queryTicket(String from, String to) {
            return new TrainTicket(from, to);
        }
    }
}

分析:上面的代码中,客户端进行查询时,系统通过TicketFactory直接创建一个火车票对象,但是这样做的话,当某个瞬间如果有大量的用户请求同一张票的信息时,系统就会创建出大量该火车票对象,系统内存压力骤增。而其实更好的做法应该是缓存该票对象,然后复用提供给其他查询请求,这样一个对象就足以支撑数以千计的查询请求,对内存完全无压力,使用 享元模式 可以很好地解决这个问题;
具体代码如下:只需对上面代码的TicketFactory进行更改,增加缓存机制:

class Client {
    public static void main(String[] args) {
        ITicket ticket = TicketFactory.queryTicket("深圳北", "潮汕");
        ticket.showInfo("硬座");
        ticket = TicketFactory.queryTicket("深圳北", "潮汕");
        ticket.showInfo("软座");
        ticket = TicketFactory.queryTicket("深圳北", "潮汕");
        ticket.showInfo("硬卧");
    }
    ...
    ...
    static class TicketFactory {
        private static Map<String, ITicket> sTicketPool = new HashMap<>();

        // 侧重于演示,不考虑性能等问题
        public static synchronized ITicket queryTicket(String from, String to) {
            String key = from + "->" + to;
            if (TicketFactory.sTicketPool.containsKey(key)) {
                System.out.println("使用缓存 ==> " + key);
                return TicketFactory.sTicketPool.get(key);
            }
            System.out.println("第一次查询,创建对象 ==> " + key);
            ITicket ticket = new TrainTicket(from, to);
            TicketFactory.sTicketPool.put(key, ticket);
            return ticket;
        }
    }
}

运行结果如下:

第一次查询,创建对象 ==> 深圳北->潮汕
深圳北->潮汕:硬座价格:429 元
使用缓存 ==> 深圳北->潮汕
深圳北->潮汕:软座价格:321 元
使用缓存 ==> 深圳北->潮汕
深圳北->潮汕:硬卧价格:481 元

可以看到,除了第一次查询创建对象后,后续查询相同车次票信息都是使用缓存对象,无需创建新对象了。

有关享元模式的更多相关文章

  1. java - dom4j库是如何实现享元模式的? - 2

    我可以在dom4j库中看到许多带有Flyweight前缀的类:FlyweightAttribute、FlyweightComment、FlyweightText等。这是java文档在FlyweightText的情况下所说的:享元文本是单链接、只读XML文本的享元模式实现。该节点可以跨文档和元素共享,但它不支持父关系。但是,我似乎无法在代码实例池中找到这些实例在文档之间共享的位置。库中是否完全实现了此功能?如果是,实现它的代码在哪里? 最佳答案 我刚刚查看了1.6.1的源代码。看起来这些Flyweight类只是原始想法的和平。至少它们

  2. c# - 了解享元模式 - 2

    Intent:Theintentofthispatternistousesharingtosupportalargenumberofobjectsthathavepartoftheirinternalstateincommonwheretheotherpartofstatecanvary.对象可以通过静态字段共享状态。使用享元模式和使用静态字段共享大量对象的内部状态有什么区别?享元通过其工厂提供的对象池是享元的真正含义吗? 最佳答案 使用静态字段,在任何一个时间点只能有一个对象实例在使用中。使用享元模式,您可以同时使用任意数量的不同

  3. php - 依赖注入(inject)和工作单元模式 - 2

    我有一个难题。我使用DI(阅读:工厂)为自制ORM提供核心组件。容器根据请求提供数据库连接、DAO、映射器及其生成的域对象。这是映射器和域对象类的基本概述classMapper{publicfunction__constructor($DAO){$this->DAO=$DAO;}publicfunctionload($id){if(isset(Monitor::members[$id]){returnMonitor::members[$id];$values=$this->DAO->selectStmt($id);//fieldmappingprocessomittedforbrevi

  4. java - 如何在 JPA 2.0 元模型中自动生成列名作为静态最终字符串? - 2

    在一些JPA注释中,我想在代码中直接使用字段名称来代替容易出错的字符串:@javax.persistence.OrderBy(value=User_.registrationDate.getName())publicListgetPlugConfigs(){...}但上面的代码不会编译,因为要获取名称,我必须使用不是常量表达式的函数(User_是JPA@StaticMetamodel生成的)。是否可以以任何方式为此使用元模型,或者我是否坚持使用直接字符串常量?有没有办法为元模型自动生成这样的字符串常量?(我正在使用maven-processor-plugin生成)

  5. java - 为什么编译器不识别元模型属性? - 2

    javase6项目是否支持eclipselinkjpa2的criteriaapi?如果没有,那是我的问题。我是否需要在persistence.xml中为条件api指定任何特殊内容?这是我的条件查询:finalEntityTypeMeaning_=em.getMetamodel().entity(Meaning.class);finalCriteriaBuildercb=em.getCriteriaBuilder();CriteriaQuerycq=cb.createQuery(Integer.class);finalRootmeng=cq.from(Meaning.class);cq.

  6. java - 那里有任何 Java 享元模式实现吗? - 2

    很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visitthehelpcenter.关闭11年前。我一直在寻找享元模式的实现,并在到达Google搜索的第20页后放弃了。虽然那里有无数愚蠢的例子,但似乎没有人发布过Java中的可重用实现。对我来说,flyweight只有在您必须保留许多这样的实例时才真正有意义,因此它必须作为一个集合来实现。我想要的是一个采用byte/short/int/longmapper实现并返回List、Set或Map的工厂,它看起来像一个普通的对象集合,但在内部存储它

  7. java - 嵌入式的 JPA 2 XML 映射,以便它与 Hibernate 元模型生成器一起工作 - 2

    我想为Hibernate(版本1.1.1-Final)(在Spring应用程序中)使用JPA2元模型生成器。因为我使用一个映射父类(superclass),它是所有实体的基础,并且这个类位于不同的jar中(为了重用)我需要在XML中显式映射这个类(仅用于元模型生成,因为它有效没有任何额外的时间)---可能有人会提示如何解决这个问题,但这不是问题。此映射的父类(superclass)(BusinessEntity)使用嵌入式类(BusinessId)。@SuppressWarnings("serial")@MappedSuperclasspublicabstractclassBusine

  8. java - 使用 Maven 生成 Hibernate 元模型中的 IllegalStateException - 2

    我正在使用hibernate-jpamodelgen通过Maven生成元模型类。当我运行mvncleanpackage时,它工作正常,没有任何问题。但是如果我第二次运行mvnpackage(没有清理),我会得到以下异常:Anexceptionhasoccurredinthecompiler(1.8.0_51).PleasefileabugattheJavaDeveloperConnection(http://java.sun.com/webapps/bugreport)aftercheckingtheBugParadeforduplicates.Includeyourprograman

  9. java - 如何使用 Gradle 5.x 生成 JPA 元模型 - 2

    我目前正在尝试从gradle4.8.1升级到5.1.1,但未能为我们的代码生成hibernate元模型。问题是gradle5忽略了通过编译类路径传递的注释处理器,但我发现的所有插件都在使用它(即“-proc:only”)。我试图明确指定注释处理器,正如gradle(https://docs.gradle.org/4.6/release-notes.html#convenient-declaration-of-annotation-processor-dependencies)所指出的那样annotationProcessor'org.hibernate:hibernate-jpamod

  10. java - Spring:缺少 JPA 元模型 - 2

    我无法理解带有JPA存储库的简单SpringMVC项目出了什么问题。能否给个提示。域:packagecom.test.app;@Entity@Table(name="foo_table")publicclassFooDomain{@Id@Column(name="id",unique=true,nullable=false)privateIntegerid;@Column(name="text",nullable=false)privateStringtext;//getters&settershere...存储库packagecom.test.app;@RepositoryDefin

随机推荐