设计模式之原型模式!附Java代码示例!

大家好,欢迎来到程序视点 我是你们的老朋友.安戈。

今天,我们一起来分享创建型模式的最后一个模式:原型模式

定义

它的定义非常简单易懂。

用原型实例指定创建对象种类,并通过拷贝原型创建新的对象

通俗点的说法就是:照着原来的实例创建一个新对象 再通俗点:克隆一个对象实例

原型模式简介

上面关于原型模式定义已经说的很明白了。在含义上没有其他需要特别说明的。这里要注意的是Java中对这种模式的处理方式。

Java 中 Object 类是所有类的根类,Object 类提供了一个 clone 方法,该方法可以将一个 Java 对象复制一份,但是需要实现 clone 的 Java 类必须要实现一个接口 Cloneable,该接口表示该类能够复制且具有复制的能力

说了这么长一串,就是告诉大家:必须实现 Cloneable 接口原型模式里有哪些角色呢?

  • 抽象原型类。它负责定义用于复制现有实例来生成新实例的方法。碰巧Java中的 Cloneable 接口就可以表示它
  • 具体原型类。它就是负责实现复制或克隆现有实例的类

示例

来看看克隆羊的例子吧!

ini 复制代码
public class Sheep implements Cloneable {  
    private String name;  
    private Integer age;  
    private String color;  

    public Sheep(String name, Integer age, String color) {  
        this.name = name;  
        this.age = age;  
        this.color = color;  
    }  

   ... setter & getter  

    @Override  
    public String toString() {  
        return "Sheep{" +  
                "name='" + name + ''' +  
                ", age=" + age +  
                ", color='" + color + ''' +  
                '}';  
    }  

    @Override  
    protected Object clone() {  
        Sheep sheep = null;  
        try {  
            sheep = (Sheep) super.clone();  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
        return sheep;  
    }  
}  

测试下:

ini 复制代码
public class Test {  
    public static void main(String[] args) {  
        Sheep sheep = new Sheep("Tom", 1, "白色");  
        for (int i = 0; i < 10; i++) {  
            // 克隆了10只羊  
            Sheep sheep1 = (Sheep) sheep.clone();  
            System.out.println(sheep1);  
        }  
    }  
}  

这里要说明两个要点:

  • 克隆对象与原对象不是同一个对象,即sheep.clone() != sheep
  • 克隆对象与原对象的类型一样(原型-->原来的类型),即sheep.clone().getClass() == sheep.getClass()

深/浅拷贝问题

所谓浅拷贝,就是对象的成员属性是引用类型时,克隆后的新对象中的成员属性引用的依旧是原对象中成员属性的内存地址。也就是说:成员属性如果是引用类型,克隆的是内存地址;这个内存地址是不变的,指向的是同一个

如果对象的成员属性是基本数据类型,那默认就是值传递。也就是将该属性值复制一份给新的对象。上面克隆羊的例子就是浅拷贝。

PS:这里有个特殊的引用类型类--String。String没有实现Cloneable,但它是"不可变的",表现出来的是"深拷贝"的特性。

所谓深拷贝,就是复制对象的所有基本数据类型的成员变量值,同时为所有引用数据类型的成员变量申请存储空间,并复制每个引用数据类型成员变量所引用的对象。也就是说:全员复制

来个深拷贝的例子作为结尾。

java 复制代码
public class Shepherd implements Cloneable {  

    private String name;  
    private int age;  
    private Sheep sheep;  

    ... setter & getter  

    @Override  
    public Object clone() throws CloneNotSupportedException {  
        Shepherd shepherd = (Shepherd) super.clone();  
        // Sheep 需要实现 Cloneable, [强调]注意: Bag 这里只有基本数据类  
        // 如果要实现完全深拷贝, Shepherd类中只能含有非基本数据类型, 如果有非基本数据类, 那么在 sheep.clone()中必须再做一次类似的深拷贝复制  
        shepherd.setSheep((Sheep) this.sheep.clone());  
        return shepherd;  
    }  
}  

重点是 Shepherd 类中重写的 clone() 方法。Shepherd shepherd = (Shepherd) super.clone();完成的就是浅拷贝。此次的 shepherd 对象中的 sheep 成员属性还引用的是原对象中的 sheep 的内存地址。 好在我们的 Sheep 也实现了 Cloneable 类。克隆一份 sheep 对象赋值回去就达到深拷贝的效果了。于是就有shepherd.setSheep((Sheep) this.sheep.clone());

PS:关于深拷贝,可以细细体会下上面的这个示例。

最后

【程序视点】助力打工人减负,从不是说说而已!

如果你觉得这篇教程有帮助,别忘了【点赞+收藏+关注】三连支持!

后续安戈会持续分享更多开发工具和技巧,敬请期待!如果有其他工具需求,欢迎留言讨论~ 🚀

相关推荐
柏油16 分钟前
MySQL InnoDB 架构
数据库·后端·mysql
一个热爱生活的普通人25 分钟前
Golang time 库深度解析:从入门到精通
后端·go
一只叫煤球的猫30 分钟前
怎么这么多StringUtils——Apache、Spring、Hutool全面对比
java·后端·性能优化
MrHuang9651 小时前
保姆级教程 | 在Ubuntu上部署Claude Code Plan Mode全过程
后端
紫穹1 小时前
008.LangChain 输出解析器
后端
维基框架1 小时前
维基框架 (Wiki FW) v1.1.1 | 企业级微服务开发框架
java·架构
小Lu的开源日常2 小时前
为什么计算机用“补码”存储整数?
设计模式·面试·计算机组成原理
某空_2 小时前
【Android】BottomSheet
java
10km2 小时前
jsqlparser(六):TablesNamesFinder 深度解析与 SQL 格式化实现
java·数据库·sql·jsqlparser
苏三说技术2 小时前
Token,Session,Cookie,JWT,Oauth2傻傻分不清楚
后端