微信搜索【三太子敖丙】关注这个贪财好色的程序员。

本文 GitHub https://github.com/JavaFamily 已收录,有一线大厂面试完整考点、资料以及我的系列文章。

前言

在设计模式的系列文章中,我们前面已经写了工厂模式单列模式建造者模式,在针对创建型模式中,今天想跟大家分享的是原型模式

其实原型模式在我们的代码中是很常见的,但是又容易被我们所忽视的一种模式,那么什么是原型模式呢?

原型模式其实就是一种克隆对象的方法,在我们的编码时候是很常见的,比如我们常用的的BeanUtils.copyProperties就是一种对象的浅copy,其实现在我们实例化对象操作并不是特别耗费性能,所以在针对一些特殊场景我们还是需要克隆那些已经实例化的对象的:

  • 依赖外部资源或硬件密集型操作,比如数据库查询,或者一些存在IO操作的场景
  • 获取相同对象在相同状态的拷贝从而不需要重复创建获取状态的操作的情况

看下我们的类图:

在上面的图中我们可以看出原型模式其实很简单:

  • 第一个是抽象原型(prototype)声明clone方法,可以是接口可以是基类,在简单的场景下我们都可以不用基类直接具体类就可以了。
  • 第二个就是具体原型类(concreteprototype)实现或者扩展clone方法,当我们在具体的原型类中的对象方法时,就会返回一个基类的抽象原型对象

针对上面理论知识,我们还是实际的举一个例子吧!

举例

假设现在我们有这么一种场景,公司搞一场活动有五万个商品参加此次活动,我们需要从后台能定时同步每个商品的销量,方便我们为后面的活动做商品分析,我们要怎么处理这个销量同步问题?

首先在这里销量和库存都是属于热点数据,但肯定都是相互隔离的因为库存是要求实时性很高的,销量可以允许有短暂延时,只要能保证数据能够最终一致性就行,所以下单的同时我们可以根据一个MQ去更新我们数据库里的商品销量。

在我们去查看销量的时候我们不能每次都是去查DB所以我们可以通过redis缓存来处理,同时我们在缓存中记录一下我们当前查询的更新时间。

再次查询时通过redis数据里面的更新时间,作为查询条件去查询DB中的更新时间大于我们当前redis中的记录时间,这样就减少了SQL的扫表的行数(更新的数据与全量数据相比,更新的数据量还是占少数的)

基于上面流程我们开始写demo了

在这里demo中我们先是创建了一个ItemSold类,以及一个SkuSold类同时ItemSold重写Cloneable里面的clone方法。然后在最后的测试类mian方法中我调用了clone方法,copy一个新的商品销量类。

细心的同学在看结果的时候不知道有没有发现一个问题?在for循环里面,我分别打印出来的ItemSold 以及 SkuSold对象他们的内存地址。

复制出来的SkuSold的内存地址居然和原型地址一样,ItemSold的复制就和原型地址不一样了,针对这个问题这里我们就要聊聊原型模式的两种实现浅拷贝深拷贝了。

这里说明一下我们在for循环里面是做数据convert,一般来说我们不会引用底层模型来做返回结果模型,需要做一层转化,来达到防腐的效果。为了体现深浅拷贝,所以写的比较简单,具体还是需要自己根据实际情况来做。

浅拷贝和深拷贝

  • 浅拷贝:当拷贝对象只包含简单的数据类型比如int、float 或者不可变的对象(字符串)时,就直接将这些字段复制到新的对象中。而引用的对象并没有复制而是将引用对象的地址复制一份给克隆对象

  • 深拷贝:不管拷贝对象里面简单数据类型还是引用对象类型都是会完全的复制一份到新的对象中

举个例子这就好比两兄弟大家买衣服可以一人一套,然后房子大家住在一套房子里(浅拷贝),当两个人成家立业了,房子分开了一人一套互不影响(深拷贝)

看完这张图,大家也就明白了,上面的demo是一个浅拷贝,那么我们要怎么做才能实现深拷贝呢?

首先我们先来看下 Java的提供的Cloneable 接口

看接口上面的解释大致可以理解为:

一个类实现了Cloneable接口,来实现这个类的clone方法从而可以合法地对该类实例进行按字段复制,假设这个类没有实现Cloneable接口的实例上调用Object的clone()方法,则会导致抛出CloneNotSupporteddException异常。

那么我们这里怎么实现深拷贝呢?

第一种:在重写ItemSold里面的clone方法时,再针对SkuSold也进行一次拷贝 (因为我们这里时List对象,只能是先拿到浅拷贝,再通过浅拷贝的List对象进行遍历再调用引用对象的clone方法来实现深拷贝)

这里如果引用对象存在多级情况下我们可能就要考虑用递归了实现,但是代码看上去就会复杂很多了。

第二种:通过序列化把对象写入流中再从流中取出来

针对上面的两种写法其实都是可以实现的,但是不管用哪种方式,深拷贝都比浅拷贝花时间和空间,所以还是酌情考虑。其实在现在已经有很多针对浅拷贝和深拷贝的工具类

  • 深拷贝(deep copy):SerializationUtils
  • 浅拷贝(shallow copy):BeanUtils

思考

针对上面的业务场景我们也可以通过其他的方式统计商品销量,可以再通过一个MQ去增加销量的同时再去更新redis缓存,但是需要我们注意的是在针对一些核心业务数据和非核心业务数据尽量不要共用一个消费者组,防止影响核心数据的消费速率。同时我们在做设计的时候多想想这么做有什么优点,又有什么缺点,开发成本问题等。

其实在其他的地方我们可以用到原型模式,比如我们在发松活动的PUSH通知,针对平台百万、千万、甚至上亿的用户发送通知的时候通知的内容基本都是一样的只是推送用户不一样或者有些特别字段值的小改动,那我们这里就可以用原型模式来做,同时开启多线程来做push,需要注意的是这里的线程安全问题,所以在每个线程内部去做copy对象。

总结

原型模式使用起来简单,但是在我们每次在clone基类或者有引用对象的时候需要我们去修改原型对象的clone方法,这不符合我们开闭原则

在一般情况下是不建议用这种模式的除非创建的对象成本特别大,或者在一些特殊场景使用,最后针对一些不常用的模式我不会详细跟大家分享,但是我会在后面做个分享总结,后面开始为大家分享行为型模式。


敖丙把自己的面试文章整理成了一本电子书,共 1630页!

干货满满,字字精髓。目录如下,还有我复习时总结的面试题以及简历模板,现在免费送给大家。

链接:https://pan.baidu.com/s/1ZQEKJBgtYle3v-1LimcSwg 密码:wjk6

我是敖丙,你知道的越多,你不知道的越多,感谢各位人才的:点赞收藏评论,我们下期见!


文章持续更新,可以微信搜一搜「 三太子敖丙 」第一时间阅读,回复【资料】有我准备的一线大厂面试资料和简历模板,本文 GitHub https://github.com/JavaFamily 已经收录,有大厂面试完整考点,欢迎Star。

设计模式 - Prototype 原型模式相关推荐

  1. 精读《设计模式 - Prototype 原型模式》

    Prototype(原型模式) Prototype(原型模式)属于创建型模式,既不是工厂也不是直接 New,而是以拷贝的方式创建对象. 意图:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的 ...

  2. C++设计模式-Prototype原型模式

    作用: 用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. Prototype模式提供了一个通过已存在对象进行新对象创建的接口(Clone), Clone()实现和具体的语言相关,在C+ ...

  3. 设计模式之原型模式(Prototype)摘录

    23种GOF设计模式一般分为三大类:创建型模式.结构型模式.行为模式. 创建型模式包括:1.FactoryMethod(工厂方法模式):2.Abstract Factory(抽象工厂模式):3.Sin ...

  4. 设计模式:原型模式(Prototype)

    欢迎支持笔者新作:<深入理解Kafka:核心设计与实践原理>和<RabbitMQ实战指南>,同时欢迎关注笔者的微信公众号:朱小厮的博客. 欢迎跳转到本文的原文链接:https: ...

  5. 乐在其中设计模式(C#) - 原型模式(Prototype Pattern)

    [索引页] [源码下载] 乐在其中设计模式(C#) - 原型模式(Prototype Pattern) 作者:webabcd 介绍 用原型实例指定创建对象的种类,并且通过拷贝这个原型来创建新的对象. ...

  6. C#设计模式(6)——原型模式(Prototype Pattern)

    一.引言 在软件系统中,当创建一个类的实例的过程很昂贵或很复杂,并且我们需要创建多个这样类的实例时,如果我们用new操作符去创建这样的类实例,这未免会增加创建类的复杂度和耗费更多的内存空间,因为这样在 ...

  7. 【设计模式】原型模式 ( 概念简介 | 使用场景 | 优缺点 | 基本用法 )

    文章目录 I . 原型模式 概念简介 II . 原型模式 使用场景 III . 原型模式 优缺点 IV . 原型模式 实现及 简单示例 I . 原型模式 概念简介 原型模式 : 用原型实例指定创建对象 ...

  8. Prototype原型模式(创建型模式)

    1.原型模式解决的问题 现在有一个抽象的游戏设施建造系统,负责构建一个现代风格和古典风格的房屋和道路. 前提:抽象变化较慢,实现变化较快(不稳定) 整个抽象的游戏设施建造系统相对变化较慢,本例中只有一 ...

  9. 【GOF23设计模式】原型模式

    [GOF23设计模式]原型模式 来源:http://www.bjsxt.com/  一.[GOF23设计模式]_原型模式.prototype.浅复制.深复制.Cloneable接口  浅复制 1 pa ...

最新文章

  1. html页面关闭前提示信息,【转】表单提交及关闭当前页面并刷新数据
  2. opencv中伪彩色applyColorMap函数(C++ / Python)
  3. oracle数据库迁移方案二
  4. 微信公众号页面模版怎么添加文章推荐功能
  5. 2017百度之星初赛B场第一题Chess--简单杨辉三角问题
  6. oracle12C推SCN,Oracle 的 DBMS_SCN 修正以及SCN的auto-rollover新特性
  7. unity透明通道加颜色_Unity的Gamma颜色空间和Linear颜色空间的小研究
  8. 扒小程序 小程序反编译 获取小程序源码 完美反编译任何小程序完整代码
  9. 软考软件测评师知识点总结
  10. 我们期待自己成为一个优秀的软件模型设计者
  11. 1024,来一套程序员续命操!
  12. 流量负载_指挥流量:揭开互联网规模负载平衡的神秘面纱
  13. 【操作系统】30天自制操作系统--(1)虚拟机加载最小操作系统
  14. 电脑裸奔-中木马-QQ被盗-大团圆结局(QQ申诉艰难旅程)
  15. 【Jenkins】的四种插件安装方式
  16. /oa/web应用程序中的服务器错误修复,如何处理OA系统在线阅读或编辑文档时weboffice控件提示“文件存取错误”的问题?...
  17. 小程序代码托管无忧,云效 代码管理接入支付宝小程序平台
  18. 用xgboost模型对特征重要性进行排序
  19. 橡胶支座抗压弹性模量计算公式_(完整版)橡胶支座计算
  20. iOS Still Image Capture Using AVCaptureSession(使用AVCaptureSession获取静止Image)

热门文章

  1. Ros 应用程序的多种调试方法
  2. 广义拉格朗日与对偶问题
  3. 精选课程 | Python数据分析实战(学术)
  4. 鸡生蛋,蛋生鸡问题。JS顶级对象Function,Object关系
  5. 我的MyBatis学习之路
  6. 从零开始学USB(十九、USB接口HID类设备(一)_HID描述符)
  7. 注意力评分函数(掩蔽softmax操作,加性注意力,缩放点积注意力)
  8. 配置Fiori for iOS开发环境
  9. Minecraft Java版1.12.2整合包崩溃求助
  10. STM32 IO口设置