给女友讲讲设计模式——原型模式(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并发编程!(8):设计模式-享元模式设计指南
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的经典对象复用设计模式-享元模式,废话不多说让我们直接开始。
116 0
|
29天前
|
存储 Java 关系型数据库
Java 项目实战基于面向对象思想的汽车租赁系统开发实例 汽车租赁系统 Java 面向对象项目实战
本文介绍基于Java面向对象编程的汽车租赁系统技术方案与应用实例,涵盖系统功能需求分析、类设计、数据库设计及具体代码实现,帮助开发者掌握Java在实际项目中的应用。
49 0
|
3月前
|
安全 Java 测试技术
Java 大学期末实操项目在线图书管理系统开发实例及关键技术解析实操项目
本项目基于Spring Boot 3.0与Java 17,实现在线图书管理系统,涵盖CRUD操作、RESTful API、安全认证及单元测试,助力学生掌握现代Java开发核心技能。
125 1
|
4月前
|
人工智能 Java 开发者
【Java实例-简易计算机】使用Java实现简单的计算机案例
一个简单的Java案例——“简易计算器”,帮助编程新手快速上手。通过实现用户输入、基本逻辑运算和结果输出,学习者可以掌握变量声明、Scanner对象使用、控制流语句等关键知识点。文章分为设计思路、关键知识点、完整代码和测试运行四个部分。
137 9
【Java实例-简易计算机】使用Java实现简单的计算机案例
|
5月前
|
Java 测试技术 项目管理
【JavaEE】从 0 到 1 掌握 Maven 构建 Java 项目核心技巧 解锁 Java 项目高效管理实用实例
本文从Maven基础概念讲起,涵盖安装配置、核心概念(如POM与依赖管理)及优化技巧。结合Java Web项目实例,演示如何用Maven构建和管理项目,解决常见问题,助你高效掌握这一强大工具,提升Java开发与项目管理能力。适合初学者及进阶开发者学习。资源链接:[点此获取](https://pan.quark.cn/s/14fcf913bae6)。
180 6
|
4月前
|
Java 开发者
【Java实例-神秘年龄】用Java挑战你的直觉
我们一起走进这款款简单却充满趣味的Java小游戏——“神秘年龄”。这款游戏不仅适合编程初学者作为练习项目,也能为有一定基础的开发者提供一个轻松的编程小憩。
49 0
【Java实例-神秘年龄】用Java挑战你的直觉
|
4月前
|
Java 开发者
【Java实例-神秘硬币】用Java投掷你的幸运硬币,你是猜正还是反?
本文分享了一个简单有趣的编程案例——猜硬币正反面游戏。通过模拟抛硬币(0为正面,1为反面),用户输入猜测值,程序判断结果并输出。
115 0
【Java实例-神秘硬币】用Java投掷你的幸运硬币,你是猜正还是反?
|
4月前
|
存储 算法 Java
【Java实例-智慧牌局】Java实现赌桌上的21点
游戏规则:游戏开始时,玩家和庄家各获得两张牌,玩家可以看到自己手中的两张牌以及庄家的一张明牌。玩家需要根据手中的牌面总和,选择“要牌”(Hit)以获取更多牌,或“停牌”(Stand)停止要牌。如果玩家的牌面总和超过21点,即为爆牌,玩家立即输掉游戏。若玩家选择停牌,庄家则开始行动,其策略是当牌面总和小于17点时必须继续要牌。若庄家牌面总和超过21点,则庄家爆牌,玩家获胜。若双方均未爆牌,最终比较牌面总和,更接近21点的一方获胜;若牌面总和相同,则游戏以平局结束。
69 0
|
4月前
|
Java 开发者
【Java实例-英雄对战】Java战斗之旅,既分胜负也决生死
游戏规则:在“英雄对战”中,玩家和敌人轮流选择行动,目标是在对方生命值归零前将其击败。游戏开始时,玩家和敌人都有100生命值。每回合,玩家可以选择“攻击”,“追击”,“闪避反击”这三种行动之一。
62 0
|
4月前
|
Java
【Java实例-小兵拆炸弹】Java打造数学挑战-拆炸弹
今天,我将向大家分享一款用Java开发的控制台小案例——“小兵拆炸弹”。游戏规则:玩家需要在有限的尝试次数内解开一系列数学题,以成功拆解炸弹。游戏的目标是连续答对五道数学题,每道题都由系统随机生成。如果玩家在五次机会内成功解密,游戏胜利;否则,炸弹爆炸,游戏结束。
79 0