Java基于享元模式实现五子棋游戏功能实例详解

本文实例讲述了Java基于享元模式实现五子棋游戏功能。分享给大家供大家参考,具体如下:

一、模式定义

享元模式,以共享的方式高效地支持大量的细粒度对象。通过复用内存中已存在的对象,降低系统创建对象实例的性能消耗。享元的英文是Flyweight,表示特别小的对象,即细粒度对象。

二、模式举例

1. 模式分析

我们借用五子棋游戏来说明这一模式。

2. 享元模式静态类图

3. 代码示例

3.1 创建抽象棋子一AbstractChessman

package com.demo.flyweight.object;
public abstract class AbstractChessman {
  // 棋子坐标
  protected int x;
  protected int y;
  // 棋子类别(黑|白)
  protected String chess;
  public AbstractChessman(String chess) {
    this.chess = chess;
  }
  // 点坐标设置
  public abstract void point(int x, int y);
  // 显示棋子信息
  public void show() {
    System.out.println(this.chess + "(" + this.x + "," + this.y + ")");
  }
}

3.2 创建黑子一BlackChessman

package com.demo.flyweight.object;
public class BlackChessman extends AbstractChessman {
  /**
   * 构造方法 初始化黑棋子
   */
  public BlackChessman() {
    super("●");
    System.out.println("--BlackChessman Construction Exec!!!");
  }
  // 点坐标设置
  @Override
  public void point(int x, int y) {
    this.x = x;
    this.y = y;
    // 显示棋子内容
    show();
  }
}

3.3 创建白子一WhiteChessman

package com.demo.flyweight.object;
public class WhiteChessman extends AbstractChessman {
  /**
   * 构造方法 初始化白棋子
   */
  public WhiteChessman() {
    super("○");
    System.out.println("--WhiteChessman Construction Exec!!!");
  }
  // 点坐标设置
  @Override
  public void point(int x, int y) {
    this.x = x;
    this.y = y;
    // 显示棋子内容
    show();
  }
}

3.4 创建棋子工厂一FiveChessmanFactory

package com.demo.flyweight.factory;
import java.util.Hashtable;
import com.demo.flyweight.object.AbstractChessman;
import com.demo.flyweight.object.BlackChessman;
import com.demo.flyweight.object.WhiteChessman;
public class FiveChessmanFactory {
  // 单例模式工厂
  private static FiveChessmanFactory fiveChessmanFactory = new FiveChessmanFactory();
  // 缓存存放共享对象
  private final Hashtable<Character, AbstractChessman> cache = new Hashtable<Character, AbstractChessman>();
  // 私有化构造方法
  private FiveChessmanFactory() {
  }
  // 获得单例工厂
  public static FiveChessmanFactory getInstance() {
    return fiveChessmanFactory;
  }
  /**
   * 根据字符获得棋子
   *
   * @param c
   *      (B:黑棋 W:白棋)
   * @return
   */
  public AbstractChessman getChessmanObject(char c) {
    // 从缓存中获得棋子对象实例
    AbstractChessman abstractChessman = this.cache.get(c);
    if (abstractChessman == null) {
      // 缓存中没有棋子对象实例信息 则创建棋子对象实例 并放入缓存
      switch (c) {
      case 'B':
        abstractChessman = new BlackChessman();
        break;
      case 'W':
        abstractChessman = new WhiteChessman();
        break;
      default:
        break;
      }
      // 为防止 非法字符的进入 返回null
      if (abstractChessman != null) {
        // 放入缓存
        this.cache.put(c, abstractChessman);
      }
    }
    // 如果缓存中存在 棋子对象则直接返回
    return abstractChessman;
  }
}

3.5 客户端实现一Client

package com.demo;
import java.util.Random;
import com.demo.flyweight.factory.FiveChessmanFactory;
import com.demo.flyweight.object.AbstractChessman;
/**
 * 主应用程序
 *
 * @author
 */
public class Client {
  /**
   * @param args
   */
  public static void main(String[] args) {
    // 创建五子棋工厂
    FiveChessmanFactory fiveChessmanFactory = FiveChessmanFactory
        .getInstance();
    Random random = new Random();
    int radom = 0;
    AbstractChessman abstractChessman = null;
    // 随机获得棋子
    for (int i = 0; i < 10; i++) {
      radom = random.nextInt(2);
      switch (radom) {
      // 获得黑棋
      case 0:
        abstractChessman = fiveChessmanFactory.getChessmanObject('B');
        break;
      // 获得白棋
      case 1:
        abstractChessman = fiveChessmanFactory.getChessmanObject('W');
        break;
      }
      if (abstractChessman != null) {
        abstractChessman.point(i, random.nextInt(15));
      }
    }
  }
}

4. 运行结果

--WhiteChessman Construction Exec!!!

○(0,2)

○(1,6)

--BlackChessman Construction Exec!!!

●(2,3)

○(3,14)

○(4,13)

○(5,8)

●(6,14)

●(7,0)

●(8,3)

○(9,8)

三、享元模式的两种状态

内蕴状态不会随环境的改变而改变,是存储在享元对象内部状态信息,困此内蕴状态是可以共享的,对于任何一个享元对象来讲,它的值是完全相同的。就像五子棋中的"黑子"和"白子",它代表的状态就是内蕴状态。

外蕴状态它会随环境的改变而改变,因此不可以共享状态,对于不同的享元对象讲,它的值可能是不同的。享元对象的外蕴状态必须由客户端保存,在享元对象被创建之后,需要使用的时候再传入享元对象内部。就像五子棋的位置信息,代表的状态就是享元对象的外蕴状态。

所以,享元的外蕴状态和内蕴状态是两类相互独立的状态,彼此没关联。

四、该模式设计原则

1. 共享细粒度对象,降低内存空间。

2. 有效地隔离系统中变化部分和不变部分。

五、使用场合

1. 当系统中某个对象类型的实例较多的时候。

2. 在系统设计中,对象实例进行分类后,发现真正有区别的分类很少的时候。

六、享元模式静态类图

更多关于java算法相关内容感兴趣的读者可查看本站专题:《Java数据结构与算法教程》、《Java操作DOM节点技巧总结》、《Java文件与目录操作技巧汇总》和《Java缓存操作技巧汇总》

希望本文所述对大家java程序设计有所帮助。

您可能感兴趣的文章:

  • Java设计模式之装饰者模式详解和代码实例
  • 举例讲解Java设计模式编程中Decorator装饰者模式的运用
  • Java装饰者模式实例详解
  • JAVA装饰者模式(从现实生活角度理解代码原理)
  • Java基于装饰者模式实现的图片工具类实例【附demo源码下载】
  • Java基于外观模式实现美食天下食谱功能实例详解
  • Java基于中介者模式实现多人聊天室功能示例
  • Java使用观察者模式实现气象局高温预警功能示例
  • Java使用访问者模式解决公司层级结构图问题详解
  • Java基于代理模式解决红酒经销问题详解
  • Java基于装饰者模式实现的染色馒头案例详解
(0)

相关推荐

  • Java基于代理模式解决红酒经销问题详解

    本文实例讲述了Java基于代理模式解决红酒经销问题.分享给大家供大家参考,具体如下: 一. 模式定义 在代理模式中,有两个对象参与处理同一请求,接收的请求由代理对象委托给真实对象处理,代理对象控制请求的访问,它在客户端应用程序与真实对象之间起到了一个中介桥梁的作用.代理模式使用对象聚合代替继承,有效地降低了软件模块之间的耦合度. 二. 模式举例 1 模式分析 我们借用红酒经销来说明这一模式. 2 代理模式静态类图 3 代码示例 3.1 红酒工厂接口一IRedWine package com.de

  • Java基于装饰者模式实现的图片工具类实例【附demo源码下载】

    本文实例讲述了Java基于装饰者模式实现的图片工具类.分享给大家供大家参考,具体如下: ImgUtil.java: /* * 装饰者模式实现图片处理工具类 * 类似java的io流 - * Img类似低级流可以独立使用 * Press和Resize类似高级流 * 需要依赖于低级流 */ package util; import java.io.File; import java.util.List; /** * 图片工具类(装饰者)和图片(被装饰者)的公共接口 * @author xlk */

  • Java装饰者模式实例详解

    本文实例讲述了Java装饰者模式.分享给大家供大家参考,具体如下: 装饰模式 在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能.它是通过创建一个包装对象,也就是装饰来包裹真实的对象. 装饰对象接受所有来自客户端的请求.它把这些请求转发给真实的对象.装饰对象可以在转发这些请求以前或以后增加一些附加功能. 这样就确保了在运行时,不用修改给定对象的结构就可以在外部增加附加的功能.在面向对象的设计中,通常是通过继承来实现对给定类的功能扩展. 优点 1. 装饰模式(Decorator)模式与

  • Java使用访问者模式解决公司层级结构图问题详解

    本文实例讲述了Java使用访问者模式解决公司层级结构图问题.分享给大家供大家参考,具体如下: 一. 模式定义 访问者模式:是表示一个作用于某对象结构中各个元素的操作,它使用户可以在不改变各元素类的前提下定义作用于这些元素的新操作. 二. 模式举例 1 模式分析 我们借用公司层级结构来说明这一模式. 2 访问者模式静态类图 3 代码示例 3.1 抽象员工一Staff package com.demo.structure; import com.demo.visitor.IVisitor; /**

  • Java使用观察者模式实现气象局高温预警功能示例

    本文实例讲述了Java使用观察者模式实现气象局高温预警功能.分享给大家供大家参考,具体如下: 一.模式定义 观察者模式,又称为发布/订阅模式.观察者模式定义了对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新. 二.模式举例 1 模式分析 我们借用气象局高温预警来说明这一模式. 2 观察者模式静态类图 3 代码示例 3.1观察者接口一IObserver package com.demo.observer; import com.demo.subje

  • Java基于装饰者模式实现的染色馒头案例详解

    本文实例讲述了Java基于装饰者模式实现的染色馒头案例.分享给大家供大家参考,具体如下: 一.模式定义 装饰者模式,是在不改变原类文件和使用继承的情况下,动态扩展一个对象功能,它是通过创建一个包装对象,也就是装饰来包装真实的对象. 装饰对象和真实对象有相同接口,这样客户端对象就可以和真实对象相同方式和装饰对象交互. 装饰对象包含一个真实对象的引用. 二.模式举例 1. 模式分析 我们借用黑心商贩制做染色馒头案例说明这一模式. 2. 装饰者模式静态类图 3. 代码示例 3.1 创建馒头接口--IB

  • Java基于中介者模式实现多人聊天室功能示例

    本文实例讲述了Java基于中介者模式实现多人聊天室功能.分享给大家供大家参考,具体如下: 一 模式定义 中介者模式,用一个中介对象来封装一系列对象之间的交互,使各个对象中不需要显示地引用其他对象实例,从而降低各个对象之间的耦合度,并且可以独立地改变对象间的交互关系. 二 模式举例 1 模式分析 我们借用多人聊天室来说明这一模式 2 中介模式静态类图 3 代码示例 3.1中介者接口--IMediator package com.demo.mediator; import com.demo.coll

  • JAVA装饰者模式(从现实生活角度理解代码原理)

    装饰者模式可以动态地给一个对象添加一些额外的职责.就增加功能来说,Decorator模式相比生成子类更为灵活. 该模式的适用环境为: (1)在不影响其他对象的情况下,以动态.透明的方式给单个对象添加职责. (2)处理那些可以撤消的职责. (3)当不能采用生成子类的方法进行扩充时.一种情况是,可能有大量独立的扩展,为支持每一种组合将产生大量的子类,使得子类数目呈爆炸性增长.另一种情况可能是因为类定义被隐藏,或类定义不能用于生成子类. 实现该模式的关键步骤: (1)Component(被装饰对象基类

  • Java基于外观模式实现美食天下食谱功能实例详解

    本文实例讲述了Java基于外观模式实现美食天下食谱功能.分享给大家供大家参考,具体如下: 一.模式定义 外观模式,是软件工程师常用的一种软件设计模式.它为子系统中的一组接口提供一个统一的高层接口,使子系统更容易使用.外观模式通过一个外观接口读/写子系统中的各接口的数据资源,而客户可以通过外观接口读取内部资源库,不与子系统产生交互. 二.模式举例 1. 模式分析 我们借用美食天下菜谱中制作糖醋排骨这一道菜来说明这一模式. 2. 外观模式静态类图 3. 代码示例 3.1 创建糖醋排骨接口一ISpar

  • 举例讲解Java设计模式编程中Decorator装饰者模式的运用

    概念 装饰者模式动态地将责任附加到对象上.若要扩展功能,装饰者提供了比继承更有弹性的替代方案. 装饰者和被装饰对象有相同的超类型. 你可以用一个或多个装饰者包装一个对象. 既然装饰者和被装饰对象有相同的超类型,所以在任何需要原始对象(被包装的)的场合 ,可以用装饰过的对象代替它. 装饰者可以在所委托被装饰者的行为之前与/或之后,加上自己的行为,以达到特定的目的. 对象可以在任何时候被装饰,所以可以在运行时动态地.不限量地用你喜欢的装饰者来装饰 对象. 在Java中,io包下的很多类就是典型的装饰

  • Java设计模式之装饰者模式详解和代码实例

    装饰者模式可以给已经存在的对象动态的添加能力.下面,我将会用一个简单的例子来演示一下如何在程序当中使用装饰者模式. 1.装饰者模式 让我们来假设一下,你正在寻找一个女朋友.有很多来自不同国家的女孩,比如:美国,中国,日本,法国等等,他们每个人都有不一样的个性和兴趣爱好,如果需要在程序当中模拟这么一种情况的话,假设每一个女孩就是一个Java类的话,那么就会有成千上万的类,这样子就会造成类的膨胀,而且这样的设计的可扩展性会比较差.因为如果我们需要一个新的女孩,就需要创建一个新的Java类,这实际上也

随机推荐