对象复用,『享元模式』

目录:设计模式之小试牛刀
源码路径:Github-Design Pattern


定义:(Flyweight Pattern)

使用共享对象可有效地支持大量的细粒度的对象。

类图:

享元模式通用类图

启示:

也许大家在平时的日常工作中,或多或少的都遇到过OutOfMemory的系统异常。而这种问题基本上都是程序中维护了太多的对象没有及时释放导致的。
解决这个问题一般有两种方式,第一种为类实现IDiposable接口,在代码中显示调用Dispose()方法,及时释放托管和非托管资源。
另一种方式,就是使用享元模式,使用共享对象池技术,减少对象的重复创建。

我们还是结合现实实例来分析一吧,先来看几张图:

阿里云的通知邮件
邮件举例1
邮件举例2

虽然是一个简单的邮件发送,但如果放到一个大的基数上,就不再是一个简简单单的邮件发送那么简单,我们需要考虑各种可能出现的性能问题。我们先抛开高可用高并发高性能这些高大上的技术不讲,先来思考下如何使用享元模式保证程序正常可用吧。

代码:

根据上面的两张邮件示例,我们可以看出邮件一般就包括收件人、发件人、主题、内容和签名。而内容又有很大的相似之处,我们可以通过模板来构造邮件内容。下面我们来抽象出一个Email类:

public class Email
{
    public string Receiver { get; set; }
    public string Sender { get; }
    public string Subject { get; }
    public string Template { get; }
    public string Signature { get; }

    public Email(string sender, string subject, string template, string signature)
    {
        Sender = sender;
        Subject = subject;
        Template = template;
        Signature = signature;
    }
}

这里面我们除了Recceiver属性暴露了Set方法外,其他属性都只能通过构造函数初始化。通过这种方式我们就可以简单的确保了对象属性(内部状态)的稳定性。
下面再来看看我们通过简单工厂来维持的对象池。

public static class EmailTemplateFactory
{
    /// <summary>
    /// 预置模板
    /// </summary>
    private static readonly Dictionary<string, string> SubjectAndContentMapping = new Dictionary<string, string>()
    {
        {
            "待修复漏洞通知",
            @"尊敬的用户:云盾检测到您的服务器存在phpwindv9任务中心GET型CSRF代码执行漏洞,
              目前已为您研发了漏洞补丁,可在云盾控制台进行一键修复。为避免该漏洞被黑客利用,
              建议您尽快修复该漏洞。您可以点击此处登录云盾 - 服务器安全(安骑士)控制台进行查看和修复"
        },
        {
            "阿里云ECS即将到期通知",
            @"您有1台云服务器ECS将于一周后正式到期。未续费的云服务器ECS实例到期后将停止服务,
              到期后数据为您保留7天,逾期未续费实例与磁盘会被释放,数据不可恢复。
              为了保证您的服务正常运行,请及时续费。"
        },
        {"阿里云故障通告", "您的服务器存在故障,请您了解!"},
        {"阿里云升级通知", "我们将对阿里云进行升级,会存在服务器短暂不可用情况,请知悉!"}
    };

    /// <summary>
    /// 定义对象池
    /// </summary>
    static readonly ConcurrentDictionary<string, Email> EmailTemplates = new ConcurrentDictionary<string, Email>();

    /// <summary>
    /// 根据主题获取模板
    /// </summary>
    /// <param name="subject"></param>
    /// <returns></returns>
    public static Email GetTemplate(string subject)
    {
        Email email = null;

        if (!EmailTemplates.ContainsKey(subject))
        {
            string template;
            SubjectAndContentMapping.TryGetValue(subject, out template);
            email = new Email("system@notice.aliyun.com", subject, string.IsNullOrWhiteSpace(template) ? subject : template, "阿里云计算公司");
            EmailTemplates.TryAdd(subject, email);
        }
        else
        {
            EmailTemplates.TryGetValue(subject, out email);
        }
        return email;
    }
}

主要定义了一个预置模板集合和一个线程安全的ConcurrentDictionary<string, Email>对象池。用户需要对象时,首先从对象池中获取,如果对象池中不存在,则创建一个新的享元对象返回给用户,并在对象池中保存该新增对象。
下一步来看一下具体的场景类:

static void Main(string[] args)
{

    for (int i = 0; i < 2000000; i++)
    {
        string receiver = $"kehu{i}@qq.com";
        //通过简单工厂维护的对象池获取已经封装好的内部状态的对象。
        var email = EmailTemplateFactory.GetTemplate("阿里云漏洞修复");
        //修改外部状态
        email.Receiver = receiver;
        SendEmail(email);
    }

    Console.ReadLine();
}

private static void SendEmail(Email email)
{
    Console.WriteLine($"主题为『{email.Subject}』的邮件已发送至:{email.Receiver}");
}

总结:

享元模式中我们要对享元对象有个清晰的认识,能够正确的分清内部和外部状态。
内部状态:在享元对象内部不随外界环境改变而改变的共享部分。
外部状态:随着环境的改变而改变,不能够共享的状态就是外部状态。
弄清了享元对象,下面一步就是构建对象池,对象池一般可以采用简单工厂模式来维护,但线程安全不可忽略。

优缺点:

享元模式与原型模式有异曲同工之妙,原型模式通过Clone的方式来重复利用已创建的对象,享元模式通过分离外部状态和内部状态来实现对象的复用,二者都能有效降低堆内存的占用的提高程序的性能。但原型模式核心在于克隆,而享元模式在于内部状态的共享,和独立的外部状态。

应用场景:

  1. 系统中存在大量相似对象,增大系统开销
  2. 对象的复用。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容

  • 1 场景问题# 1.1 加入权限控制## 考虑这样一个问题,给系统加入权限控制,这基本上是所有的应用系统都有的功能...
    七寸知架构阅读 2,488评论 1 57
  • 定义 Flyweight在拳击比赛中指最轻量级,即“蝇量级”或“雨量级”。这里选择使用“享元模式”的意译,是因为这...
    步积阅读 1,612评论 0 2
  • 原文地址:LoveDev 享元模式(Flyweight Pattern):运用共享技术有效地支持大量细粒度对象的复...
    KevinLive阅读 300评论 0 0
  • 我把所有的不安搬到我自己的房间锁紧门以防它们的气味飘走 我坐在阁楼的窗边看你你 自由的女孩在齐膝的杂草中如小狗一样...
    陈果_周绿阅读 275评论 1 2
  • 昨天由上海浦东飞往兰州,在下午两点多将要降落的时候,广播来了:由于雷雨,无法降落,飞机将在上空盘旋等待天气转好。大...
    老撒阅读 315评论 0 3