给女友讲讲设计模式——原型模式(JAVA实例)4

简介: 前言上高中那阵子,特别迷恋火影,所以每周都会等着更新。最让我难忘的是里面的主人公鸣人,和他的各种各样炫酷的技能。他的影分身,一下子可以分离出来多个自己,并且这些自己的副本使用的技能都是相同的。

前言

上高中那阵子,特别迷恋火影,所以每周都会等着更新。最让我难忘的是里面的主人公鸣人,和他的各种各样炫酷的技能。他的影分身,一下子可以分离出来多个自己,并且这些自己的副本使用的技能都是相同的。如果他的副本想要使用其他的技能,除非分身再克隆出来一个技能的副本,然后去进行学习。

正文

刚刚说的那个例子,大家先记着,接下来我慢慢揭开圆形模式的面纱。
原型模式,简单来说就像是我们操纵电脑使的复制粘贴,只不过我们是要去操纵对象。在java中使用原型模式特别的简单,只需要重写clone()方法,然后实现Cloneable接口(其实这个接口是一个空接口,只是为了告诉jvm,这个类是可以克隆的);
原型模式一共分为两种:
1.浅克隆
2.深克隆
下面看看浅克隆的代码
这是一个忍者的实体类,代表鸣人的。

package prototype;

public class Nanjia implements Cloneable{
    
    private String name;
    private Integer age;
    private Skill skill;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    
    
    public Skill getSkill() {
        return skill;
    }
    public void setSkill(Skill skill) {
        this.skill = skill;
    }
    @Override
    public Nanjia clone() {
        try {
            Nanjia nanjia=(Nanjia)super.clone();
            return nanjia;
        } catch (CloneNotSupportedException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return null;
    }
    @Override
    public String toString() {
        return "Nanjia [name=" + name + ", age=" + age + ", skill=" + skill + "]";
    }
    
}

然后是一个skill的实体类,可以装者技能的种类和名称

package prototype;

public class Skill{
    private String name;
    private String type;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getType() {
        return type;
    }
    public void setType(String type) {
        this.type = type;
    }
    @Override
    public String toString() {
        return "Skill [name=" + name + ", type=" + type + "]";
    }
    
}

package prototype;

public class MainTest {

    public static void main(String[] args) {
        Nanjia nanjia=new Nanjia();
        nanjia.setAge(16);
        nanjia.setName("漩涡鸣人");
        Skill skill=new Skill();
        skill.setName("仙人模式");
        nanjia.setSkill(skill);
        System.out.println("这是需要被克隆的类");
        System.out.println(nanjia);
        Nanjia nanjia2=nanjia.clone();
        Skill skill2=nanjia2.getSkill();
        skill2.setName("螺旋丸");
        nanjia2.setSkill(skill2);
        System.out.println("这是克隆出来的类");
        System.out.println(nanjia2);
        System.out.println("这是原来的类");
        System.out.println(nanjia);
    }
}

在测试类中我们做了这样的操作,将一些属性赋予nanji的实体类中,然后我们通过克隆复制出了一个副本,可以看作类似于影分身一样,然后拿出副本中的技能将它转换为别的技能,最后我们会发现原来本身的技能也发生了变化,这是怎么一回事呢?
这是因为浅克隆,在克隆的过程中,对象中的实体引用,只是复制了这个引用的地址,所以在修改的时候就相当于把原来本体的技能同样进行修改。
那我们要是想不影响之前的本体该怎么办呢,接下来隆重介绍深克隆
深克隆是把忍者类中的技能类也重写了克隆方法,这样在影分身的副本中我们就可以自由转换,而不影响忍者这个本体了。
然后是深克隆的代码:

package prototype;

public class Nanjia implements Cloneable{
    
    private String name;
    private Integer age;
    private Skill skill;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    
    
    public Skill getSkill() {
        return skill;
    }
    public void setSkill(Skill skill) {
        this.skill = skill;
    }
    @Override
    public Nanjia clone() {
        try {
            Nanjia nanjia=(Nanjia)super.clone();
            //关键的地方在这里,我们像克隆忍者一样,把技能这个类也进行了克隆
            nanjia.setSkill((Skill)nanjia.getSkill().clone());
            return nanjia;
        } catch (CloneNotSupportedException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return null;
    }
    @Override
    public String toString() {
        return "Nanjia [name=" + name + ", age=" + age + ", skill=" + skill + "]";
    }
    
}

我们继承了Cloneable接口,重写了Clone这个方法。

package prototype;

public class Skill implements Cloneable{
    private String name;
    private String type;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getType() {
        return type;
    }
    public void setType(String type) {
        this.type = type;
    }
    @Override
    public String toString() {
        return "Skill [name=" + name + ", type=" + type + "]";
    }
    
    @Override
    public Object clone() {
        try {
            return super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
            return null;
        }
    }
    
}

package prototype;

public class MainTest {

    public static void main(String[] args) {
        Nanjia nanjia=new Nanjia();
        nanjia.setAge(16);
        nanjia.setName("漩涡鸣人");
        Skill skill=new Skill();
        skill.setName("仙人模式");
        nanjia.setSkill(skill);
        System.out.println("这是需要被克隆的类");
        System.out.println(nanjia);
        Nanjia nanjia2=nanjia.clone();
        Skill skill2=nanjia2.getSkill();
        skill2.setName("螺旋丸");
        nanjia2.setSkill(skill2);
        System.out.println("这是克隆出来的类");
        System.out.println(nanjia2);
        System.out.println("这是原来的类");
        System.out.println(nanjia);
    }
}

后记

克隆模式却是在制作副本,并且副本极其相似的情况下,给我们带来了很多便利性,但是如果类结构相对复杂,我们不但得在每一个实体中加入克隆方法,并且这样做还不符合开闭原则。所以在使用之前,应该根据需求,看看需不需要用到原型模式

目录
相关文章
|
5月前
|
设计模式 网络协议 数据可视化
Java 设计模式之状态模式:让对象的行为随状态优雅变化
状态模式通过封装对象的状态,使行为随状态变化而改变。以订单为例,将待支付、已支付等状态独立成类,消除冗长条件判断,提升代码可维护性与扩展性,适用于状态多、转换复杂的场景。
755 157
|
5月前
|
设计模式 Java Spring
Java 设计模式之责任链模式:优雅处理请求的艺术
责任链模式通过构建处理者链,使请求沿链传递直至被处理,实现发送者与接收者的解耦。适用于审批流程、日志处理等多级处理场景,提升系统灵活性与可扩展性。
613 2
|
7月前
|
设计模式 缓存 Java
Java设计模式(二):观察者模式与装饰器模式
本文深入讲解观察者模式与装饰器模式的核心概念及实现方式,涵盖从基础理论到实战应用的全面内容。观察者模式实现对象间松耦合通信,适用于事件通知机制;装饰器模式通过组合方式动态扩展对象功能,避免子类爆炸。文章通过Java示例展示两者在GUI、IO流、Web中间件等场景的应用,并提供常见陷阱与面试高频问题解析,助你写出灵活、可维护的代码。
|
5月前
|
设计模式 算法 搜索推荐
Java 设计模式之策略模式:灵活切换算法的艺术
策略模式通过封装不同算法并实现灵活切换,将算法与使用解耦。以支付为例,微信、支付宝等支付方式作为独立策略,购物车根据选择调用对应支付逻辑,提升代码可维护性与扩展性,避免冗长条件判断,符合开闭原则。
1054 35
|
5月前
|
设计模式 消息中间件 传感器
Java 设计模式之观察者模式:构建松耦合的事件响应系统
观察者模式是Java中常用的行为型设计模式,用于构建松耦合的事件响应系统。当一个对象状态改变时,所有依赖它的观察者将自动收到通知并更新。该模式通过抽象耦合实现发布-订阅机制,广泛应用于GUI事件处理、消息通知、数据监控等场景,具有良好的可扩展性和维护性。
467 8
|
6月前
|
存储 Java 关系型数据库
Java 项目实战基于面向对象思想的汽车租赁系统开发实例 汽车租赁系统 Java 面向对象项目实战
本文介绍基于Java面向对象编程的汽车租赁系统技术方案与应用实例,涵盖系统功能需求分析、类设计、数据库设计及具体代码实现,帮助开发者掌握Java在实际项目中的应用。
258 0
|
设计模式 缓存 安全
Java设计模式的单例模式应用场景
Java设计模式的单例模式应用场景
368 4
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
227 4
|
设计模式 存储 负载均衡
【五】设计模式~~~创建型模式~~~单例模式(Java)
文章详细介绍了单例模式(Singleton Pattern),这是一种确保一个类只有一个实例,并提供全局访问点的设计模式。文中通过Windows任务管理器的例子阐述了单例模式的动机,解释了如何通过私有构造函数、静态私有成员变量和公有静态方法实现单例模式。接着,通过负载均衡器的案例展示了单例模式的应用,并讨论了单例模式的优点、缺点以及适用场景。最后,文章还探讨了饿汉式和懒汉式单例的实现方式及其比较。
【五】设计模式~~~创建型模式~~~单例模式(Java)
|
设计模式 安全 Java
Java 编程中的设计模式:单例模式的深度解析
【9月更文挑战第22天】在Java的世界里,单例模式就像是一位老练的舞者,轻盈地穿梭在对象创建的舞台上。它确保了一个类仅有一个实例,并提供全局访问点。这不仅仅是代码优雅的体现,更是资源管理的高手。我们将一起探索单例模式的奥秘,从基础实现到高级应用,再到它与现代Java版本的舞蹈,让我们揭开单例模式的面纱,一探究竟。
133 11