Java基于享元模式实现五子棋游戏功能实例详解
作者:chengqiuming 发布时间:2023-07-23 16:20:18
本文实例讲述了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程序设计有所帮助。
来源:https://blog.csdn.net/chengqiuming/article/details/70139362


猜你喜欢
- 一:简述我们很多时候为了实现数据在线程级别下的隔离,会使用到ThreadLocal,那么TheadLocal是如何实现数据隔离的呢?今天就和
- 目录第一点——加载网页结构第二点——解析网页结构,转变为符合需求的数据实体第三点——保存数据实体,转变为符合需求的数据实体第四点——main
- 有时候我们做Android开发,需要弹一个用户提示,但是有时候设计的提示弹窗是带有图片的,我们每次写一个特别麻烦。所以我特地封装了一个工具类
- 话不多说,请看代码:using System;using System.Web;using System.Drawing;using Sys
- 描述说明:public class TryCatchStu { /*try catch:自己处理异常 *t
- 一、单例模式的思想想整理一些 java 并发相关的知识,不知道从哪开始,想起了单例模式中要考虑的线程安全,就从单例模式开始吧。以前写过单例模
- 一:背景1. 讲故事高级语言玩多了,可能很多人对指针或者汇编都淡忘了,本篇就和大家聊一聊指针,虽然C#中是不提倡使用的,但你能说指针在C#中
- 一、什么是JMSJMS即Java消息服务(Java Message Service)应用程序接口,是一个Java平台中关于面向消息中间件(M
- 多态是同一个行为具有多个不同表现形式或形态的能力。多态性意味着有多重形式。在面向对象编程范式中,多态性往往表现为"一个接口,多个功
- 前言在上一篇文章中讲完了SpringMVC的大部分知识,此篇文章中主要讲解 * 。上一篇文章🚩 * 的使用是非常普遍的。例如在 OA系统中通
- 在网上找了半天,说的都没有解决我的问题,我自己花了点时间在idea中找到并解决了问题,希望可以帮助到大家。File---->setti
- 使用lzma SDK对7z文件简单解压缩有时候我们只需要单纯对lzma算法压缩的7z文件进行解压,有时需要在嵌入式设备上解压,使用p7zip
- 本文实例讲述了Android编程实现使用webView打开本地html文件的方法。分享给大家供大家参考,具体如下:在布局的配置文件里:<
- Android通过访问网页查看网页源码1.添加网络权限<!--访问网络的权限--> <uses-permission an
- 本文实例为大家分享了Android开发之自定义闹钟实现,供大家参考,具体内容如下闹钟时间设置及显示闹钟的布局很简单,就是一个简单时间设置,所
- 这个小代码是我凭自己对指针和链表的理解和认识,自己实现的,没有参考其他人的代码,如果有相同的地方,那真的只是巧合,代码我在ubuntu 15
- 一、背景说明由于以前在项目中一直使用sqlmap.xml进行mybatis语句的编写和实现,其xml实现动态更新和查询较为方便,而目前由于技
- 前排提示,我在这个工具类加了@Component注解,如果在springboot的项目使用,记得通过@Autowired注入使用。impor
- springboot去除控制台打印的debug日志1.创建logback-spring.xml文件文件内容如下<?xml versio
- idea删除模块后重新创建显示该模块已经被注册原因:注册信息没有删除干净解决方案:找到gradle.xml,modules.xml,work