23种设计模式之原型模式

目录

  • [1. 简介](#1. 简介)
  • [2. 实现](#2. 实现)
    • [2.1 原来的方式(使用newg关键字)](#2.1 原来的方式(使用newg关键字))
    • [2.2 使用接口Cloneable](#2.2 使用接口Cloneable)
    • [2.3 浅拷贝与深拷贝(拓展)](#2.3 浅拷贝与深拷贝(拓展))
  • [3. 总结](#3. 总结)

1. 简介

原型模式 (Prototype Pattern)是一种创建型设计模式。它的主要思想是通过复制一个已经存在的对象(原型)来创建新的对象,而不是使用传统的通过构造函数来创建对象的方式。这种模式的好处是可以在运行时动态地创建对象,并且可以根据需要灵活地修改复制后的对象。

例如,在一个游戏中,可能有多种类型的怪物。当需要生成新的怪物时,不是每次都从无到有地创建,而是复制一个已有的怪物原型,然后根据具体情况(如等级、属性加成等)进行修改,这样可以提高效率。

在 Java 中,实现原型模式通常需要让原型类实现java.lang.Cloneable接口。这个接口是一个标记接口,它没有任何方法,只是用来表明这个类的对象是可以被克隆的。

然后,在原型类中重写Object类的clone()方法。clone()方法是Object类提供的一个受保护的方法,用于创建并返回对象的一个副本。

2. 实现

2.1 原来的方式(使用newg关键字)

Pig.java

java 复制代码
public class Pig {
    private String name;
    private String doSomeThing;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getDoSomeThing() {
        return doSomeThing;
    }

    public void setDoSomeThing(String doSomeThing) {
        this.doSomeThing = doSomeThing;
    }

    @Override
    public String toString() {
        return "Pig{" +
                "name='" + name + '\'' +
                ", doSomeThing='" + doSomeThing + '\'' +
                "} , " + super.toString();
    }
}

Test.java

java 复制代码
public class Test {
    public static void main(String[] args) {
        Pig pig = new Pig();
        pig.setName("小猪佩奇");
        pig.setDoSomeThing("吃饭");
        System.out.println(pig);

        Pig pig2 = new Pig();
        pig2.setName("小猪乔治");
        pig2.setDoSomeThing("吃米");
        System.out.println(pig2);

        Pig pig3 = new Pig();
        pig2.setName("大猪");
        pig2.setDoSomeThing("睡觉");
        System.out.println(pig2);
    }
}

输出结果:

Pig{name='小猪佩奇', doSomeThing='吃饭'} , yxz.prototype.Pig@1b6d3586
Pig{name='小猪乔治', doSomeThing='吃米'} , yxz.prototype.Pig@4554617c
Pig{name='大猪', doSomeThing='睡觉'} , yxz.prototype.Pig@4554617c

2.2 使用接口Cloneable

Pig.java

java 复制代码
package yxz.prototype;


public class Pig implements Cloneable{
    public Pig(){
        System.out.println("小猪初始化。。。");
    }

    private String name;
    private String doSomeThing;

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getDoSomeThing() {
        return doSomeThing;
    }

    public void setDoSomeThing(String doSomeThing) {
        this.doSomeThing = doSomeThing;
    }

    @Override
    public String toString() {
        return "Pig{" +
                "name='" + name + '\'' +
                ", doSomeThing='" + doSomeThing + '\'' +
                "} , " + super.toString();
    }

}

Test.java

java 复制代码
package yxz.prototype;

public class Test {
    public static void main(String[] args) throws CloneNotSupportedException {
        Pig pig = new Pig();
        pig.setName("小猪佩奇");
        pig.setDoSomeThing("吃饭");
        System.out.println(pig);

        Pig pig2 = new Pig();
        pig2.setName("小猪乔治");
        pig2.setDoSomeThing("吃米");
        System.out.println(pig2);

        Pig pig3 = new Pig();
        pig2.setName("大猪");
        pig2.setDoSomeThing("睡觉");
        System.out.println(pig2);

        // 使用原型设计模式
        Pig peiqi = new Pig();
        peiqi.setName("小猪佩奇");
        peiqi.setDoSomeThing("吃饭");
        System.out.println(peiqi);

        Pig george = (Pig)peiqi.clone();
        george.setName("小猪乔治");
        george.setDoSomeThing("吃米");
        System.out.println(george);

    }
}

运行结果:

小猪初始化。。。
Pig{name='小猪佩奇', doSomeThing='吃饭'} , yxz.prototype.Pig@1b6d3586
小猪初始化。。。
Pig{name='小猪乔治', doSomeThing='吃米'} , yxz.prototype.Pig@4554617c
小猪初始化。。。
Pig{name='大猪', doSomeThing='睡觉'} , yxz.prototype.Pig@4554617c
小猪初始化。。。
Pig{name='小猪佩奇', doSomeThing='吃饭'} , yxz.prototype.Pig@74a14482
Pig{name='小猪乔治', doSomeThing='吃米'} , yxz.prototype.Pig@1540e19d

2.3 浅拷贝与深拷贝(拓展)

浅拷贝 :在上面的示例中,clone()方法实现的是浅拷贝。浅拷贝会创建一个新的对象,新对象的基本数据类型的属性会被复制,但是如果属性是引用类型,那么只是复制了引用,而不是复制引用指向的对象。例如,如果Prototype类中有一个引用类型的属性AnotherClass anotherObject,在浅拷贝后,新对象和原对象的anotherObject属性将指向同一个对象。

深拷贝:深拷贝会创建一个新的对象,并且会递归地复制对象的所有属性,包括引用类型的属性所指向的对象。这样,原始对象和克隆后的对象完全独立,修改其中一个不会影响另一个。

深拷贝的方式:

java 复制代码
class MyClass implements Cloneable {
    private String field1;
    private NestedClass nestedObject;

    @Override
    protected Object clone() throws CloneNotSupportedException {
        MyClass cloned = (MyClass) super.clone();
        cloned.nestedObject = (NestedClass) nestedObject.clone(); // 深拷贝内部的引用对象
        return cloned;
    }
}

可以看到,深拷贝是新创建了一个对象。

3. 总结

  • 应用场景
    • 对象的创建成本较高时:如果对象的创建过程涉及到复杂的初始化操作,如读取配置文件、建立数据库连接等,使用原型模式可以避免重复这些操作。通过复制已经初始化好的原型对象,可以快速地创建新的对象。
    • 动态配置对象时:在需要根据运行时的情况动态地生成对象的场景中,原型模式很有用。例如,在图形编辑软件中,用户可以复制已有的图形对象(如圆形、矩形等),然后根据自己的需求修改复制后的图形的属性,如大小、颜色等。
相关推荐
春风十里不如你95272 分钟前
【设计模式】【行为型模式(Behavioral Patterns)】之命令模式(Command Pattern)
设计模式·命令模式
Clockwiseee3 小时前
JS原型、原型链以及原型链污染学习
javascript·学习·原型模式
十五年专注C++开发3 小时前
C++中的链式操作原理与应用(一)
开发语言·c++·设计模式
旺代6 小时前
C++设计模式(单例模式)
c++·单例模式·设计模式
LightOfNight6 小时前
【设计模式】创建型模式之单例模式(饿汉式 懒汉式 Golang实现)
单例模式·设计模式·golang
那年星空15 小时前
Flutter 设计模式全面解析:抽象工厂
flutter·设计模式·架构
博风16 小时前
设计模式:10、外观模式
设计模式·外观模式
你好helloworld18 小时前
设计模式之观察者模式
观察者模式·设计模式
zzzhpzhpzzz1 天前
设计模式——抽象工厂模式
设计模式·抽象工厂模式