享元(Flyweight)模式:通过共享技术以便有效的支持大量细粒度的对象。
享元模式在阎宏的《java与模式》中分为单纯享元模式和复合享元模式,复合模式的复合享元是不可以共享的,享元对象能做到共享的关键是区分内蕴态(Internal State)和外蕴态( External State)。这两个“蕴态”翻译的太难懂,我不是说翻译的不好,可能是我理解能力差,还是《Design Pattern Elements of Reusable Object-Oriented Software》的翻译版《设计模式可复用面向对象软件的基础》一书总翻译为内部对象和外部对象,相对直白,对概念性的东西文学气味太强了就觉得很别扭。这里的角色也采用《设计模式可复用面向对象软件的基础》的说法,不区分单纯模式和复合模式,而是有一个UnSharedConcreteFlyweight(在《java与模式》里称复合享元,指明复合享元不能共享),我们这里称它不可以共享享元角色,这样享元模式的角色有:
抽象享元(Flyweight)角色:是给实现享元提供的接口。
具体享元(ConcreteFlyweight)角色:实现抽象角色,此对象必须是共享的,所含的状态必须是内部状态。
不共享享元(UnSharedConcreteFlyweight)角色:此对象不可共享,不是所有实现抽象享元接口的的对象都要共享,此对象通常将ConcreteFlyweight作为组成元素。
享元工厂(FlyweightFactory)角色:负责创建和管理享元角色,确保合理共享。
客户端(Client)角色:维持一个Flyweight对象的引用,计算或存储一个(多个)外部存储状态。享元模式的类的机构图如下:
享元模式在java.lang.String设计上的使用,我们知道java中字符串始终保持共享一份,如下面代码片段:
String m = "a";
String n = "a";
System.out.println(m==n);
这样会输出true,说明m和n指向了同一个实例,内存中也只有一个"a"。这就是享元模式在String上的使用。
享元模式在文字编辑存贮过程中的使用,这里假定文章由行对象组成,行对象由若干个字符对象组成,但是如果每个字符都保存自己的对象,那么一篇文章成千上万个字符对象,这样严重消耗系统内存,造成不可接受的运行时开销,好的方法是利用享元模式,只保存ASCII字符编码值,作为内部不变的状态,对当个字符对象进行共享,而相对字符颜色、大小这样的格式化数据作为外部状态,由客户端维护,运行时由外部传入即可。每个行作为不可共享享元对象,它是由享元对象(字符对象)组合而成的。
举例代码如下:
package flyWeight; /** * *作者:alaric *时间:2013-7-27下午4:52:41 *描述:抽象享元 */ public interface Glyph { public void draw(Context context); }
package flyWeight; /** * *作者:alaric *时间:2013-7-27下午4:53:12 *描述:具体享元 */ public class Character implements Glyph { private char c; private int size; @Override public void draw(Context context) { // TODO Auto-generated method stub this.size = context.getSize(); System.out.println(size+"号"+c+"被画出!"); } public char getC() { return c; } public void setC(char c) { this.c = c; } public Character(char c) { super(); this.c = c; System.out.println(c+"被创建!"); } }
package flyWeight; import java.util.ArrayList; import java.util.List; /** * *作者:alaric *时间:2013-7-27下午4:52:26 *描述:行 不可共享享元 */ public class Row implements Glyph { private List<Character> list = new ArrayList<>(); @Override public void draw(Context context) { } public Row() { } public void setCharacter(Glyph r){ list.add((Character) r); } public int getSize(){ return list.size(); } public String getRow(){ StringBuilder sb = new StringBuilder(); for(Character g:list){ sb.append(g.getC()); } return sb.toString(); } }
package flyWeight; import java.util.HashMap; import java.util.Map; /** * *作者:alaric *时间:2013-7-27下午4:52:08 *描述:享元工厂 */ public class GlyphFactory { private Map<String,Glyph> map = new HashMap<>(); public Glyph getGlyph(Context context){ String cStr = context.getC()+""; Glyph gl = map.get(cStr); if(gl == null){ gl = new Character(context.getC()); map.put(cStr, gl); } gl.draw(context); return gl; } }
package flyWeight; /** * *作者:alaric *时间:2013-7-27下午3:34:57 *描述:数据类 */ public class Context { private int size; private char c; public int getSize() { return size; } public void setSize(int size) { this.size = size; } public char getC() { return c; } public void setC(char c) { this.c = c; } public Context(int size, char c) { super(); this.size = size; this.c = c; } }
package flyWeight; /** * *作者:alaric *时间:2013-7-27下午4:56:01 *描述:客户端 为了简单 就直接写main方法里的 */ public class Client { /** *作者:alaric *时间:2013-7-27下午4:20:08 *描述:测试 */ public static void main(String[] args) { Row r =new Row(); GlyphFactory factory = new GlyphFactory(); Context context1= new Context(12, 'a'); Glyph gly1 = factory.getGlyph(context1); r.setCharacter(gly1); Context context2= new Context(13, 'a'); Glyph gly2 = factory.getGlyph(context2); r.setCharacter(gly2); Context context3= new Context(13, 'b'); Glyph gly3 = factory.getGlyph(context3); r.setCharacter(gly3); System.out.println(r.getRow()); } }
运行结果:
a被创建!
12号a被画出!
13号a被画出!
b被创建!
13号b被画出!
aab
可以看出a创建了一次 ,a的大小12 ,13是外部状态所以是外部传入,外部状态不能在享元内保存,而a字符是内部状态,行Row是有字符Character 组成,Row虽然实现了抽象享元接口,但是并没有再工厂中体现共享,因为他是不可共享的享元。
设计模式系列目录:
相关推荐
装饰器模式是面向对象设计模式的一种,主要用于在不改变原有对象结构的情况下,动态地为对象增加新的功能。这种模式在Java中尤其常见,因为它允许我们遵循“开闭原则”——对扩展开放,对修改关闭。 装饰器模式的...
总之,JAVA设计模式中的组合模式提供了一种优雅的方式来处理对象的树形结构,使得我们可以在处理单个对象和整个集合时保持代码的简洁和一致性。通过理解和熟练运用这种模式,开发者可以更好地设计和维护复杂的软件...
以上只是部分Java设计模式的介绍,实际的“DesignPatterns”压缩包可能包含了这些模式的源代码实现,通过阅读和学习这些代码,开发者可以更好地理解和运用设计模式,提升代码质量。同时,结合博主提供的博客链接,...
本资料“《java设计模式》课后习题模拟试题解答——刘伟.zip”主要涵盖了Java设计模式的学习与应用,特别是针对刘伟教授的相关课程的课后习题及模拟试题的解答。 设计模式分为三大类:创建型、结构型和行为型模式。...
以上两种创建型模式,即工厂方法模式和抽象工厂模式,是Java设计模式中非常重要的一部分,它们可以帮助开发者更好地管理对象的创建过程,提高系统的灵活性和可扩展性。接下来的部分将继续探讨其他的设计模式。
Java全能学习面试手册——Java面试题库.zip 01 7道消息队列ActiveMQ面试题!.pdf 02 10道Java高级必备的Netty面试题!.pdf 03 10道Java面试必备的设计模式面试题!.pdf 04 10个Java经典的List面试题!.pdf 05 10个...
### 浅析Java设计模式【3】——代理 #### 一、代理模式概述 代理模式是一种行为型设计模式,主要用于在客户端与目标对象之间起到一个中介的作用,通过代理对象来控制对目标对象的访问。代理模式的核心在于它可以...
java设计模式——创建模式、结构模式、行为模式
### Java设计模式学习笔记——外观模式(Facade Pattern) #### 概述 设计模式是软件工程领域中一种解决常见问题的可复用解决方案。在Java开发过程中,掌握设计模式能够帮助开发者更好地组织代码结构,提高代码的...
网易云课堂微专业课程,Java核心设计模式——DAO模式的教学视频
通过深入学习《设计模式——Java语言中的应用》,开发者能够更好地理解和应用这些模式,从而编写出更加高效、可扩展的Java程序。无论是初级开发者还是经验丰富的程序员,都应该不断探索和实践设计模式,以提升自己的...
首先,我们来看"Java设计模式——观察者模式的两种情况": 1. **主动通知**: 在这种情况下,被观察者对象在自身状态发生变化时,会主动调用`Observable`接口提供的`notifyObservers()`方法,将变化通知给所有注册的...
在Java编程语言中,集合和三层设计模式是两个非常重要的概念,对于开发高效、可维护的软件系统至关重要。本文将详细探讨这两个主题。 首先,我们来了解Java中的集合。Java集合框架是Java SE API的一个核心部分,它...
2. **MVC模式**:MVC(Model-View-Controller)模式是Java Web开发中的常用设计模式,用于分离业务逻辑、数据模型与用户界面。我们会探讨Spring MVC框架的应用,它是Java Web中实现MVC模式的重要工具。 3. **...
设计模式是软件工程中的一种重要概念,它代表了在特定情境下解决常见问题的最佳实践。刘伟先生在讲解设计模式时,通常会深入浅出地介绍这些模式的...学习设计模式不仅可以提升个人技能,也有利于团队间的沟通和协作。
策略模式是一种行为设计模式,它使你能在运行时改变对象的行为。在Java中,策略模式主要通过定义一系列的算法,并将每一个算法封装起来,使它们可以互相替换,让算法独立于使用它的客户而变化。这种模式的核心是策略...
### Java设计模式经典教程知识点概览 #### 一、设计模式概述 设计模式是一种软件设计方法,它为软件开发者提供了一种标准化的方式去解决常见的软件设计问题。设计模式的使用可以提高代码的可读性和可维护性,同时...