深入理解Java设计模式之原型模式

 更新时间:2021年11月07日 15:04:29   作者:一指流砂~  
这篇文章主要介绍了JAVA设计模式之原型模式的的相关资料,文中示例代码非常详细,供大家参考和学习,感兴趣的朋友可以了解下

一、前言

单例模式可以避免重复创建消耗资源的对象,但是却不得不共用对象。若是对象本身也不让随意访问修改时,怎么办?通常做法是备份到副本,其它对象操作副本,最后获取权限合并,类似git上的PR操作。

二、什么是原型模式

原型模式用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象。需要注意的关键字是,新的对象,类没变。.NET在System命名空间中提供了Cloneable接口,其中它提供唯一的方法Clone(),只需要实现这个接口就可以完成原型模式了。由于它直接操作内存中的二进制流,当大量操作或操作复杂对象时,性能优势将会很明显。

三、原型模式的适用场景

多用于创建大对象,或初始化繁琐的对象。如游戏中的背景,地图。web中的画布等等

以下场景适用:

一是类初始化需要消化非常多的资源,这个资源包括数据、硬件资源等;

二是通过 new 产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式;

三是一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,可以考虑使用原型模式拷贝多个对象供调用者使用。

在实际项目中,原型模式很少单独出现,一般是和工厂方法模式一起出现,通过 clone的方法创建一个对象,然后由工厂方法提供给调用者。

四、原型模式的实现

以简历的复印来举例

1.浅拷贝实现

定义工作经历类

/// <summary>
/// 工作经历类
/// </summary>
public class WorkExperience
{
    private string _workDate;
    public string WorkDate
    {
        get { return _workDate; }
        set { _workDate = value; }
    }
     private string _company;
    public string Company
    {
        get { return _company; }
        set { _company = value; }
    }
}

定义简历类

/// <summary>
/// 简历类
/// </summary>
class Resume : ICloneable
{
    private string name;
    private string sex;
    private string age;
     private WorkExperience work;
     public Resume(string name)
    {
        this.name = name;
        work = new WorkExperience();
    }
     /// <summary>
    /// 设置个人信息
    /// </summary>
    /// <param name="sex"></param>
    /// <param name="age"></param>
    public void SetPersonalInfo(string sex, string age)
    {
        this.sex = sex;
        this.age = age;
    }
     /// <summary>
    /// 设置工作经历
    /// </summary>
    /// <param name="workDate"></param>
    /// <param name="company"></param>
    public void SetWorkExperience(string workDate, string company)
    {
        work.WorkDate = workDate;
        work.Company = company;
    }
     /// <summary>
    /// 显示
    /// </summary>
    public void Display()
    {
        Console.WriteLine("{0}{1}{2}", name, sex, age);
        Console.WriteLine("工作经历:{0}{1}", work.WorkDate, work.Company);
    }
     public object Clone()
    {
        //创建当前object的浅表副本
        return (object)this.MemberwiseClone();
    }
}

客户端调用

static void Main(string[] args)
{
    Resume a = new Resume("张三");
    a.SetPersonalInfo("男", "30");
    a.SetWorkExperience("2010-2018", "腾讯公司");
     Resume b = (Resume)a.Clone();
    b.SetWorkExperience("2010-2015", "阿里公司");
     Resume c = (Resume)a.Clone();
    c.SetPersonalInfo("女", "18");  c.SetWorkExperience("2010-2015", "百度公司");
     a.Display();
    b.Display();
    c.Display();
     Console.Read();
}

结果

张三 男 30
工作经历 2010-2018 腾讯公司
张三 男 30
工作经历 2010-2018 腾讯公司
张三 女 18
工作经历 2010-2018 腾讯公司

被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用都仍然指向原来的对象,这就是浅复制。但是我们可能需要这样一种需求,要把复制的对象所引用的对象都复制一遍。比如刚才的例子,我希望a、b、c三个引用的对象都是不同的。复制时就一变二,二变三。此时,我们就要用的方式叫“深复制”

2.深拷贝实现

深复制把引用对象的变量指向复制过的新对象,而不是原来被引用的对象

/// <summary>
/// 工作经历类
/// </summary>
public class WorkExperience:ICloneable
{
    private string _workDate;
    public string WorkDate
    {
        get { return _workDate; }
        set { _workDate = value; }
    }
     private string _company;
    public string Company
    {
        get { return _company; }
        set { _company = value; }
    }
     public object Clone()
    {
        //创建当前object的浅表副本
        return (object)this.MemberwiseClone();
    }
}
/// <summary>
/// 简历类
/// </summary>
class Resume : ICloneable
{
    private string name;
    private string sex;
    private string age;
     private WorkExperience work;
     public Resume(string name)
    {
        this.name = name;
        work = new WorkExperience();
    }
     private Resume(WorkExperience work)
    {
        //提供Clone方法调用的私有构造函数,以便克隆“工作经历”数据
        this.work = (WorkExperience)work.Clone();
    }
     /// <summary>
    /// 设置个人信息
    /// </summary>
    /// <param name="sex"></param>
    /// <param name="age"></param>
    public void SetPersonalInfo(string sex, string age)
    {
        this.sex = sex;
        this.age = age;
    }
     /// <summary>
    /// 设置工作经历
    /// </summary>
    /// <param name="workDate"></param>
    /// <param name="company"></param>
    public void SetWorkExperience(string workDate, string company)
    {
        work.WorkDate = workDate;
        work.Company = company;
    }
     /// <summary>
    /// 显示
    /// </summary>
    public void Display()
    {
        Console.WriteLine("{0}{1}{2}", name, sex, age);
        Console.WriteLine("工作经历:{0}{1}", work.WorkDate, work.Company);
    }
     public object Clone()
    {
        //调用私有的构造方法,让“工作经历”克隆完成,然后再给这个简历对象的相关字段赋值,
        //最终返回一个深复制的简历对象
        Resume obj = new Resume(this.work);
        obj.name = this.name;
        obj.sex = this.sex;
        obj.age = this.age;
        return obj;
    }
}

客户端调用代码一样

结果

张三 男 30
工作经历 2010-2018 腾讯公司
张三 男 30
工作经历 2010-2015 阿里公司
张三 女 18
工作经历 2010-2015 百度公司

由于在一些特定场合,会经常涉及深复制和浅复制,比如说,数据集对象DataSet,它就有Clone()方法和Copy()方法,Clone()方法用来复制DataSet的结构,但不复制DataSet的数据,实现了原型模式的浅复制,

Copy()方法不但复制结构,还复制数据,其实就是实现了原型模式的深复制。

五、总结

原型模式通过Object的clone()方法实现,由于是内存操作,无视构造方法和访问权限,直接获取新的对象。但对于引用类型,需使用深拷贝,其它浅拷贝即可。

相关文章

  • 一文搞懂Java正则表达式的使用

    一文搞懂Java正则表达式的使用

    正则表达式,又称规则表达式,是一种文本模式。正则表达式使用单个字符串来描述、匹配一系列匹配某个句法规则的字符串,通常被用来检索、替换那些符合某个模式(规则)的文本。本文将通过示例为大家详细说说Java正则表达式的使用,感兴趣的可以了解一下
    2022-08-08
  • SpringBoot集成FTP与SFTP连接池流程

    SpringBoot集成FTP与SFTP连接池流程

    在项目开发中,一般文件存储很少再使用SFTP服务,但是也不排除合作伙伴使用SFTP来存储项目中的文件或者通过SFTP来实现文件数据的交互,这篇文章主要介绍了SpringBoot集成FTP与SFTP连接池
    2022-12-12
  • 持久层ORM框架Hibernate框架的使用及搭建方式

    持久层ORM框架Hibernate框架的使用及搭建方式

    Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,使得Java程序员可以随心所欲的使用对象编程思维来操纵数据库,本文重点给大家介绍持久层ORM框架Hibernate框架的使用及搭建方式,感兴趣的朋友一起看看吧
    2021-11-11
  • MyBatis加解密插件的示例详解

    MyBatis加解密插件的示例详解

    本文介绍了使用 MyBatis 插件实现数据库字段加解密的探索过程,实际开发过程中需要注意的细节比较多,整个流程下来我对 MyBatis 的理解也加深了,对MyBatis加解密插件感兴趣的朋友跟随小编一起看看吧
    2022-08-08
  • springboot中的pom文件 project报错问题

    springboot中的pom文件 project报错问题

    这篇文章主要介绍了springboot中的pom文件 project报错问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-01-01
  • mybatis plus自动生成器解析(及遇到的坑)

    mybatis plus自动生成器解析(及遇到的坑)

    这篇文章主要介绍了mybatis-plus自动生成器及遇到的坑,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • SpringBoot使用工具类实现获取容器中的Bean

    SpringBoot使用工具类实现获取容器中的Bean

    这篇文章主要为大家详细介绍了SpringBoot如何使用工具类实现获取容器中的Bean,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-03-03
  • 基于strict-origin-when-cross-origin问题的解决

    基于strict-origin-when-cross-origin问题的解决

    这篇文章主要介绍了基于strict-origin-when-cross-origin问题的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-03-03
  • 详解Jackson的基本用法

    详解Jackson的基本用法

    Java生态圈中有很多处理JSON和XML格式化的类库,Jackson是其中比较著名的一个。虽然JDK自带了XML处理类库,但是相对来说比较低级,使用本文介绍的Jackson等高级类库处理起来会方便很多
    2021-06-06
  • Java开发实例之图书管理系统的实现

    Java开发实例之图书管理系统的实现

    图书管理的功能大体包括:增加书籍、借阅书籍、删除书籍、查看书籍列表、退出系统、查找书籍、返还书籍这些,本文主要给大家介绍该系统的数据库语句,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-10-10

最新评论