JAVA设计模式(三)-原型

JAVA设计模式(三)-原型

本篇文章主要讲下java 创建型设计模式中的原型模式.

何谓原型模式: 简单来说就是 将一个对象作为原型,通过对其进行复制而克隆出多个和原型类似的新实例。

使用原型模式,就可以简化实例化的过程, 不必依赖于构造函数或者new关键字.

由于java 提供了clone方法, 原型设计模式的实现就很简单了.

原型模式的要素:

  1. 原型接口(Prototype Interface): 定义了克隆方法的接口.该方法用于复制现有对象并创建新对象。
  2. 原型类(Concrete Prototype Class): 实现了克隆方法,来复制自身

1: 原型类

具体的实现如下:

java 复制代码
package com.zh.xpose;

import java.util.List;

/**
 * @Author: zh
 * @Time: 23-12-22.
 * @Email:
 * @Describe:
 */
public class ConcretePrototype implements Cloneable{
    private String name;
    private List<String> fields;

    public ConcretePrototype(String name,List<String> fields) {
        System.out.println("原型创建成功");
        this.name =name;
        this.fields =fields;
    }

    public String getName() {
        return name;
    }

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

    public List<String> getFields() {
        return fields;
    }

    public void setFields(List<String> fields) {
        this.fields = fields;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        System.out.println("原型复制成功");
        return (ConcretePrototype)super.clone();
    }
}
java 复制代码
public class JavaTest {
    public static void main(String[] args) {		
		ArrayList<String> list = new ArrayList<>();
        list.add("age");
        list.add("sex");
        ConcretePrototype concretePrototype = new ConcretePrototype("person",list);
        ConcretePrototype clone = null;
        try {
            clone= (ConcretePrototype) concretePrototype.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        System.out.println(concretePrototype==clone);
        System.out.println(concretePrototype.getName() == clone.getName());
        System.out.println(concretePrototype.getFields() == clone.getFields());
        
    }
}

输出如下:

java 复制代码
exclude patterns:
原型创建成功
原型复制成功
false
true
true

2: 浅克隆和深克隆

浅克隆只会复制原型对象本身,而不会复制它所引用的对象。也就是说,克隆对象和原型对象会共享引用对象。如果原型对象中的引用对象发生改变,克隆对象中的引用对象也会发生改变。

深克隆会复制原型对象以及它所引用的对象。也就是说,克隆对象和原型对象拥有各自独立的引用对象。无论原型对象中的引用对象是否发生改变,克隆对象中的引用对象都不会受到影响。

关于浅克隆:

验证如下:

java 复制代码
		concretePrototype.setName("test111");
        list.add("第三个了");
        concretePrototype.setFields(list);
        System.out.println(concretePrototype.getName()+"  "+clone.getName());
        System.out.println(concretePrototype.getFields().size()+"  "+clone.getFields().size());

可以看到 当原型类中的list 发生变更时, clone同时发生了变更.

java 复制代码
test111  person
3  3

实现深克隆的方式有两种: 通过实现Cloneable接口和通过序列化.

这里通过修改上面的实现类中的clone方法,来实现深克隆.

java 复制代码
package com.zh.xpose;

import java.util.ArrayList;
import java.util.List;

/**
 * @Author: zh
 * @Time: 23-12-22.
 * @Email:
 * @Describe:
 */
public class ConcretePrototype implements Cloneable{
    private String name;
    private List<String> fields;

    public ConcretePrototype(String name,List<String> fields) {
        System.out.println("原型创建成功");
        this.name =name;
        this.fields =fields;
    }

    public String getName() {
        return name;
    }

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

    public List<String> getFields() {
        return fields;
    }

    public void setFields(List<String> fields) {
        this.fields = fields;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        System.out.println("原型复制成功");
        ConcretePrototype clone = (ConcretePrototype) super.clone();
        clone.setName(new String(name));
        clone.setFields(new ArrayList<>(fields));
        return clone;
    }
}

重新执行验证代码 ,结果输出如下:

arduino 复制代码
原型创建成功
原型复制成功
false
false
false
test111  person
3  2
相关推荐
不修×蝙蝠6 分钟前
SpringBoot(一)--搭建架构5种方法
java·spring boot·架构·配置·搭建
FreemanGordon1 小时前
Java volatile 关键字
java
北京_宏哥1 小时前
《手把手教你》系列基础篇(九十三)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-上篇(详解教程)
java·前端·selenium
北京_宏哥1 小时前
《手把手教你》系列基础篇(九十二)-java+ selenium自动化测试-框架设计基础-POM设计模式简介(详解教程)
java·selenium·前端工程化
当归10241 小时前
微服务与消息队列RabbitMQ
java·微服务
Lx3521 小时前
《从头开始学java,一天一个知识点》之:循环结构:for与while循环的使用场景
java·后端
Cache技术分享1 小时前
15. Java 如何声明一个变量来引用数组
java·前端
雷渊1 小时前
深入分析理解mysql的MVCC
java·数据库·面试
知其然亦知其所以然1 小时前
Java 高级面试题:Lock 到底比 synchronized 强在哪?
java·后端·面试
风象南1 小时前
Spring Boot 的 20个实用技巧
java·spring boot