首页 > 其他分享 >设计模式(十一)享元

设计模式(十一)享元

时间:2023-11-07 14:13:16浏览次数:33  
标签:享元 十一 Coordinates new 设计模式 IgoChessman public 享元类

一、定义

运用共享技术有效地支持大量细粒度对象的复用,享元模式是一种结构型模式。

二、描述

享元模式要求能够共享的对象必须是细粒度对象,因此它又称为轻量级模式。享元模式的结构较为复杂,一般结合工厂模式一起使用,在其结构图中包含了一个享元工厂类,包含以下四个角色:
1、Flyweight(抽象享元类):它通常是一个接口或抽象类,在抽象享元类中声明了具体享元类的公共方法,这些方法可以向外界提供享元对象的内部数据(内部状态),同时也可以通过这些方法设置外部数据(外部状态)
2、ConcreteFlyweight(具体享元类):它实现了抽象享元类,其实例称为享元对象,并在具体享元类中为内部提供了存储空间。通常可以结合单例模式来设计具体享元类,为每一个具体享元类提供唯一的享元对象
3、UnsharedConcreteFlyweight(非共享具体享元类):并不是所有的抽象享元类的子类都需要被共享,用户可以将不能被共享的子类可设计为非共享具体享元类,当需要一个非共享具体享元类的对象时可以直接通过实例化创建。
4、FlyweightFactory(享元工厂类):用于创建并管理享元对象,它针对抽象享元类编程,将各种类型的具体享元对象存储在一个享元池中,一般设计为一个存储键值对的集合(也可以是其他类型的集合),可以结合工厂模式设计。当用户请求一个具体享元对象时,享元工厂提供一个存储在享元池中已创建的实例或者创建一个新的实例(如果不存在),返回新创建的实例并将其存储在享元池中。

三、例子

X公司欲开发一个围棋软件,通过分析,发现在围棋棋盘中包含大量的黑子和白子,它们的形状、大小都一模一样,只是出现的位置不同而已。如果将每一个棋子都作为一个独立的对象存储在内存中,将可能导致该围棋软件在运行时所需要的内存空间较大。用享元模式来设计该软件,降低运行代价、提高系统性能。
IgoChessman:抽象享元类

public abstract class IgoChessman
{
    public abstract string GetColor();

    public void Display(Coordinates coord)
    {
        Console.WriteLine("棋子颜色:{0},棋子位置:{1}", GetColor(), coord.X + "," + coord.Y);
    }
}

/// <summary>
/// 外部状态:棋子坐标
/// </summary>
public class Coordinates
{
    public int X { get; set; }
    public int Y { get; set; }

    public Coordinates()
    {

    }

    public Coordinates(int x, int y)
    {
        this.X = x;
        this.Y = y;
    }
}

BlackIgoChessman、WhiteIgoChessman:黑棋、白棋享元类,充当具体享元类

public class BlackIgoChessman : IgoChessman
{
    public override string GetColor()
    {
        return "黑色";
    }
}

public class WhiteIgoChessman : IgoChessman
{
    public override string GetColor()
    {
        return "白色";
    }
}

IgoChessmanFactory:享元工厂类

public class IgoChessmanFactory
{
    private static readonly IgoChessmanFactory instance = new IgoChessmanFactory(); // 使用单例模式实现享元
    private static Hashtable ht;    // 使用Hashtable来存储享元对象,充当享元池

    private IgoChessmanFactory()
    {
        ht = new Hashtable();
        IgoChessman blackChess = new BlackIgoChessman();
        ht.Add("b", blackChess);
        IgoChessman whiteChess = new WhiteIgoChessman();
        ht.Add("w", whiteChess);
    }

    public static IgoChessmanFactory GetInstance()
    {
        return instance;
    }

    public IgoChessman GetIgoChessman(string color)
    {
        IgoChessman chess = ht[color] as IgoChessman;
        return chess;
    }
}

Program:客户端测试类

// 获取享元工厂
IgoChessmanFactory chessFactory = IgoChessmanFactory.GetInstance();
// 通过享元工厂获取3颗黑子
IgoChessman blackChess1 = chessFactory.GetIgoChessman("b");
IgoChessman blackChess2 = chessFactory.GetIgoChessman("b");
IgoChessman blackChess3 = chessFactory.GetIgoChessman("b");

Console.WriteLine("判断两颗黑子是否相同:{0}", object.ReferenceEquals(blackChess1, blackChess2));
// 通过享元工厂获取2颗白子
IgoChessman whiteChess1 = chessFactory.GetIgoChessman("w");
IgoChessman whiteChess2 = chessFactory.GetIgoChessman("w");

Console.WriteLine("判断两颗白子是否相同:{0}", object.ReferenceEquals(whiteChess1, whiteChess2));
// 显示棋子
blackChess1.Display(new Coordinates(1,2));
blackChess2.Display(new Coordinates(3, 4));
blackChess3.Display(new Coordinates(1, 3));
whiteChess1.Display(new Coordinates(2, 5));
whiteChess2.Display(new Coordinates(2, 4));

四、总结

1、优点

(1)可以极大减少内存中对象的数量,使得相同或相似对象在内存中只有一份,节省系统资源,提高系统性能。
(2)外部状态相对独立,不会影响内部状态,使享元对象可以在不同的环境中被共享。

2、缺点

(1)系统变的复杂,需要分离内外部状态,使程序的逻辑复杂化。
(2)为了使对象可以共享,享元模式需要将享元对象的部分状态外部化,而读取外部状态将使得运行时间变长。

标签:享元,十一,Coordinates,new,设计模式,IgoChessman,public,享元类
From: https://www.cnblogs.com/WinterSir/p/17434019.html

相关文章

  • 设计模式---策略模式+工厂
    关键词:设计模式,策略模式,工厂模式概要现在我需要实现一个功能,是添加一路SDI输出,但是输出的协议有不同,有udp、srt等,针对不同的协议我要做不同的操作,后面还有可能添加其他的协议,因此这里面用策略模式不错。由于单纯的策略模式并不能完全消除if...else...,这里我们用了工厂模式再进......
  • Oracle中B-tree索引的访问方法(十一)-- 索引的分裂行为
    索引的分裂行为当某个索引块中要插入新的索引条目,但其中又没有可用空间时,就会发生索引的分裂。根据分裂发生所在的索引块类型的不同,可以分为在根块上发生的分裂,在分支块上发生的分裂和在叶子块上发生的分裂。下面,就这三种情况做分别介绍。从前面的实验中,我们已经看到,大约每个索引块......
  • 精通协程的必会十一个高级技巧
    在Android应用开发中,协程已经成为异步编程的首选工具之一。它使并发任务管理变得更加容易,但它的强大功能远不止于此。在本文中,我们将探讨协程的高级技巧,帮助您更好地处理复杂的并发需求,提高性能和可维护性。介绍协程是Kotlin的一项强大特性,它使并发编程更加直观、简单。它允许我们......
  • 设计模式实验8,9 适配器模式,桥接模式
    适配器模式:实现一个双向适配器,使得猫可以学狗叫,狗可以学猫抓老鼠。桥接模式用桥接模式实现在路上开车这个问题,其中,车可以是car或bus,路可以是水泥路或沥青路。 ......
  • 设计模式—结构型模式之桥接模式
    设计模式—结构型模式之桥接模式将抽象与实现解耦,使两者都可以独立变化。在现实生活中,某些类具有两个或多个维度的变化,如图形既可按形状分,又可按颜色分。如何设计类似于Photoshop这样的软件,能画不同形状和不同颜色的图形呢?如果用继承方式,m种形状和n种颜色的图形就有m×n......
  • (十一)Python之字符串类型
    字符串类型Python中的字符串用单引号(‘’)或双引号(”“)括起来,同时使用反斜杠(\)转义特殊字符语法:s=”a1a2...an“(n>=0)Python使用单引号(‘)、双引号(“)、三引号(”“”)来表示字符串、其中三引号可以由多行组成,它是编写多行文本的快捷语法,常用于文档字符串,在文件的特定地点,被当作注......
  • Java设计模式之桥接模式
    桥接(Bridge)模式的定义如下:将抽象与实现分离,使它们可以独立变化。它是用组合关系代替继承关系来实现,从而降低了抽象和实现这两个可变维度的耦合度。假如你有一个几何形状(Shape)类,从它能扩展出两个子类:圆形(Circle)和方形(Square)。你希望对这样的类层次结构进行扩展以使其包含颜色,所......
  • java设计模式之适配器模式
    定义将一个接口转换成客户希望的另一个接口,使接口不兼容的那些类可以一起工作,其别名为包装器(Wrapper)。适配器模式既可以作为类结构型模式,也可以作为对象结构型模式。在适配器模式中,我们通过增加一个新的适配器类来解决接口不兼容的问题,使得原本没有任何关系的类可以协同工作。......
  • 设计模式六大原则
    前言什么是设计模式?设计模式是软件设计人员、软件开发人员在程序代码编写中总结出来的一套编码规范,设计模式起一个指导作用,用来指导我们写出高内聚低耦合,具有良好的可扩展性和可维护性的代码。为什么要学设计模式?当然,设计模式不是非学不可,不了解设计模式一样可以在工作中写出......
  • 【设计模式】策略模式在项目中的实战运用
    目录前言思考实现落地小结前言思考随着业务需求不断迭代更新,系统逻辑越来越复杂。ifelse堆砌让人眼花缭乱。那么此时就可以考虑使用设计模式,重构代码逻辑采用什么设计模式,或者哪几种设计模式组合,与实际业务场景、逻辑有关系。以下面这个场景为例:现在要将一批货物从A地点运往B地......