23模式---原型模式(浅拷贝和深拷贝)

原型模式是一种创建型设计模式,Prototype模式允许一个对象再创建另外一个可定制的对象,根本无需知道任何如何创建的细节,工作原理是:通过将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝它们自己来实施创建。

简单的说就是对象本身提供了一个可复制(克隆)的接口,用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。

这个就有两个疑问了,返回一个对象?

  • 不是有单例模式吗?

    因为单例模式返回的对象的同一个,而有些场景下虽然使用相同属性值的类,但是再某个方法中出现变化的时候,不希望影响其它对的属性值。举一个不巧当的例子,比如双胞胎出生后很多信息都相同,都是未婚,可以说属性值一样,但是有一天老二结婚了媳妇是小红,不能说老大立马也结婚,媳妇也是小红吧?简单的说就是单例模式无法满足所有遇见的问题。

  • 不是有工厂模式吗?

    举一个例子那就是抽象工厂类,以小米手机为抽象接口,然后生长的手机有不同的型号A,B,C。然后对用的工厂类有A,B,C等工厂。所以只要创建一个新的对象,就需要有新的工厂类,这个是工厂模式的优势,也是其也会让代码逻辑变复杂。

    而克隆模式就是通过一个实例小米手手机对象克隆出需要的手机对象,然后根据ABC型号,来换对应ABC属性值即可。

    • 这又产生了一个问题:既然通过实例克隆,然后再修改属性,那么我干嘛不直接通过new来实现,而通过克隆呢?不是故意让代码变得更繁琐吗?

      不是的, 因为某些复杂的对象实例的时候拷贝(克隆)的效率一般对构造的效率要高,但是也不是所有的都是快的,比如简单逻辑拷贝(克隆)也就没有直接new的效率高。

因为java中自动拷贝功能,所以先试一下克隆对比构造的创建实例的效率:

先来一个简单逻辑的:

public class test implements Cloneable {

    public static void main(String[] args) throws CloneNotSupportedException {
        Long start1= System.currentTimeMillis();
        Phone p=new Phone("小米",1999,"red");
        for (int i = 0; i <9999999 ; i++) {
            new Phone(p.getBrandname(),p.price,p.getColor());
        }
        Long end1= System.currentTimeMillis();
        System.out.println("new创建1万实例:"+ (end1-start1));

        Long start2= System.currentTimeMillis();
        Phone p2=new Phone("小米",1999,"red");
        for (int i = 0; i <9999999 ; i++) {
            p2.clone();
        }
        Long end2= System.currentTimeMillis();
        System.out.println("原型创建1万实例:"+ (end2-start2));


    }

}

class Phone implements Cloneable{
    String brandname;
    double price;
    String color;
    public Phone() {

    }
    public Phone(String brandname, double price, String color) {
        this.brandname = brandname;
        this.price = price;
        this.color = color;
    }

    public String getBrandname() {
        return brandname;
    }

    public double getPrice() {
        return price;
    }

    public String getColor() {
        return color;
    }



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

在这里插入图片描述

故意来一个在构造对象的时候调用一个方法:


public class test implements Cloneable {

    public static void main(String[] args) throws CloneNotSupportedException {
        Long start1= System.currentTimeMillis();
        Phone p=new Phone("小米",1999,"red");
        for (int i = 0; i <9999999 ; i++) {
            new Phone(p.getBrandname(),p.price,p.getColor());
        }
        Long end1= System.currentTimeMillis();
        System.out.println("new创建1万实例:"+ (end1-start1));

        Long start2= System.currentTimeMillis();
        Phone p2=new Phone("小米",1999,"red");
        for (int i = 0; i <9999999 ; i++) {
            p2.clone();
        }
        Long end2= System.currentTimeMillis();
        System.out.println("原型创建1万实例:"+ (end2-start2));


    }

}

class Phone implements Cloneable{
    String brandname;
    double price;
    String color;
    public Phone() {

    }
    public Phone(String brandname, double price, String color) {
        this.brandname = brandname;
        this.price = getRealpice(price);
        this.color = color;
    }
    public double getRealpice(double price){
        for (int i = 0; i < 5000; i++) {
            new String();

        }
        return  price;
    }
    public String getBrandname() {
        return brandname;
    }

    public double getPrice() {
        return price;
    }

    public String getColor() {
        return color;
    }



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

在这里插入图片描述

所以:只有再较为复杂的逻辑下,复制(克隆)方式的效率才会高于通过构造函数创建的对象。

Cloneable接口使用细节

java本身就支持原型模式(对象自带克隆方法),所以不需要我们手动去写如何实现这个克隆方法,所以现在看一使用细节。

先看一下Cloneable这个接口:

在这里插入图片描述

看了一下其本身没有定义任何方法,但是为什么还要写实现这个接口呢,而类可以种的clone方法显示其为重写方法,说明其在父类,所以看一些Object类

在这里插入图片描述

看英文说明,指数可以知道:

  • 此方法是native修饰,说明是系统方法,其具体实现逻辑是C写的底层实现的。其应该是直接在内存堆种进行复制新对象的。
  • 为什么要重写conle方法,而通过super调用因为其protected修饰,只能通子类来通过spuer来调用。
  • 返回的对象和被克隆的对象其是两个不同的对象,两者不是同一个对象,但是是同一个class类实例化的。
  • 虽然Cloneable没有写方法,但是如果不写,调用重写后的clone方法就会报错。

还有一点这里没有说到的,那就是clnoe方法克隆的对象是只能克隆基本数据类型,如果引用类型的属性,还是共有的,这个就又分浅拷贝和深拷贝了。(这个JavaScript种也聊过这个)

当然如果通过clnoe方法自己写行不行比如:

class Phone implements Cloneable{
    String brandname;
    double price;
    String color;
    public Phone() {

    }
    public Phone(String brandname, double price, String color) {
        this.brandname = brandname;
        this.price = price;
        this.color = color;
    }

    public String getBrandname() {
        return brandname;
    }

    public double getPrice() {
        return price;
    }

    public String getColor() {
        return color;
    }

    public void setBrandname(String brandname) {
        this.brandname = brandname;
    }

    public void setPrice(double price) {
        this.price = price;
    }

    public void setColor(String color) {
        this.color = color;
    }
// 自己这样写
    public Object clone1()  {
        Phone p=new Phone();
        p.setBrandname(this.getBrandname());
        p.setPrice(this.getPrice());
        p.setColor(this.getColor());
        return p;
    }
}

是否可以实现呢?当然也是可以的,不过创建一个对象都写一个,如果属性由几十个的话,是否就很麻烦,所以人家已经写好了,干嘛不直接用呢?不过说句实话,自己这样写有可能效率比调用系统的方法更快,毕竟所有的属性都知道得,不用通过反射再创建了。

浅拷贝

看一下java种通过实现cloneable接口实现的拷贝为什么说是浅拷贝。

public class test implements Cloneable {

    public static void main(String[] args) throws CloneNotSupportedException {
        String[] arr = {"老头环", "奎爷", "神秘臂力"};
        TestObject testObject = new TestObject("张三", Arrays.asList(arr));
        System.out.println("testObject原本数据:" + testObject.toString());
        TestObject testObject1 = (TestObject) testObject.clone();
        testObject1.setName("李四");
        System.out.println("testObject的数据:   " + testObject.toString());
        System.out.println("testObject1的数据:  " + testObject1.toString());

        Collections.reverse(testObject.arr);
        System.out.println("testObject的数据:   " + testObject.toString());
        System.out.println("testObject1的数据:  " + testObject1.toString());


    }

}

class TestObject implements Cloneable {
    String name;
    List<String> arr;

    public TestObject(String name, List<String> arr) {
        this.name = name;
        this.arr = arr;
    }

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

    public void setArr(List<String> arr) {
        this.arr = arr;
    }

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

    @Override
    public String toString() {
        return "TestObject{" +
                "name='" + name + '\'' +
                ", arr=" + Arrays.toString(arr.toArray()) +
                '}';
    }
}

在这里插入图片描述

可以看出克隆的如果是基本数据类型或者String的话克隆后的对象,不会影响彼此,但是如果是引用数据类型的话,就会影响彼此。所以一般的时候除非确定属性不是引起类型,不然不要如此使用。

深拷贝

深拷贝的的也可以通过Cloneable实现,但是先声明其由一定的局限性。

方式1:

public class test   {

    public static void main(String[] args) throws CloneNotSupportedException {

        TestObjectA testObject = new TestObjectA("张三", new TestObjectB());
        System.out.println("testObject原本数据:" + testObject.toString());
        TestObjectA testObject1 = (TestObjectA) testObject.clone();

        System.out.println("testObject的数据:   " + testObject.toString());
        System.out.println("testObject1的数据:  " + testObject1.toString());

    }

}

class TestObjectA implements Cloneable {
    String name;
    TestObjectB  testObjectB;

    public TestObjectA(String name, TestObjectB  testObjectB) {
        this.name = name;
        this.testObjectB = testObjectB;
    }

    @Override
    public String toString() {
        return "TestObjectA{" +
                " testObjectB=" + testObjectB.hashCode() +
                '}';
    }
    @Override
    public Object clone() throws CloneNotSupportedException {
        TestObjectA deep=null;
        deep= (TestObjectA) super.clone();
        deep.testObjectB= (TestObjectB) this.testObjectB.clone();
        return deep;
    }
}
class TestObjectB implements Cloneable{
    @Override
    public Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

在这里插入图片描述

这样似乎解决了浅拷贝的问题,但是其有一个弱点,那就是如果TestObjectB的属性也引用了对象,或者说TestObjectA都有多个引用属性呢?是不是代码写起来就麻烦了很多。

不过借鉴一下我再JavaScript种通过Json和对象的转换来解决深拷贝的问题。当然java种除非调用第三方的json包才可以,不过通过这个想法可以相到一件事情,那就是可以通过序列化来实现这个问题。

public class test  {

    public static void main(String[] args) throws CloneNotSupportedException {

        TestObjectA testObject = new TestObjectA("张三", new TestObjectB());
        System.out.println("testObject原本数据:" + testObject.toString());
        TestObjectA testObject1 = (TestObjectA) testObject.clone();

        System.out.println("testObject的数据:   " + testObject.toString());
        System.out.println("testObject1的数据:  " + testObject1.toString());
//        Json

    }

}

class TestObjectA implements Serializable {
    String name;
    TestObjectB testObjectB;

    public TestObjectA(String name, TestObjectB testObjectB) {
        this.name = name;
        this.testObjectB = testObjectB;
    }

    @Override
    public String toString() {
        return "TestObjectA{" +
                " testObjectB=" + testObjectB.hashCode() +
                '}';
    }

    @Override
    public Object clone() throws CloneNotSupportedException {
        ByteArrayOutputStream bos = null;
        ObjectOutputStream oos = null;
        ByteArrayInputStream bis = null;
        ObjectInputStream ois = null;
        Object object = null;
        try {
            bos = new ByteArrayOutputStream();
            oos = new ObjectOutputStream(bos);
            oos.writeObject(this);

            bis = new ByteArrayInputStream(bos.toByteArray());
            ois = new ObjectInputStream(bis);
            object = ois.readObject();

            ois.close();
            bis.close();
            oos.close();
            bos.close();

            
        } catch (Exception e) {
            System.out.println(e);
        }

        return object;
    }
}

class TestObjectB implements Serializable{

}

在这里插入图片描述

而这种方式才是再java种一般解决深拷贝问题的方式。

总结

  • 优势:

    性能提高。 逃避构造函数的约束。

  • 缺点

    配备克隆方法需要对类的功能进行通盘考虑,这对于全新的类不是很难,但对于已有的类不一定很容易,特别当一个类引用不支持串行化的间接对象,或者引用含有循环结构的时候。 必须实现 Cloneable 接口或者Serializable接口。

  • 用场景:

    • 1、资源优化场景。
    • 2、类初始化需要消化非常多的资源,这个资源包括数据、硬件资源等。
    • 3、性能和安全要求的场景。
    • 4、通过 new 产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式。
    • 5、一个对象多个修改者的场景。
    • 6、一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,可以考虑使用原型模式拷贝多个对象供调用者使用。
    • 7、在实际项目中,原型模式很少单独出现,一般是和工厂方法模式一起出现,通过 clone 的方法创建一个对象,然后由工厂方法提供给调用者。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1) 优秀的程序应该是这样的:阅读时,感觉很优雅;新增功能时,感觉很轻松;运行时,感觉很快速,这就需要设计模式支撑。2) 设计模式包含了大量的编程思想,讲授和真正掌握并不容易,网上的设计模式课程不少,大多讲解的比较晦涩,没有真实的应用场景和框架源码支撑,学习后,只知其形,不知其神。就会造成这样结果: 知道各种设计模式,但是不知道怎么使用到真实项目。本课程针对上述问题,有针对性的进行了升级 (1) 授课方式采用 图解+框架源码分析的方式,让课程生动有趣好理解 (2) 系统全面的讲解了设计模式,包括 设计模式七大原则、UML类图-类的六大关系、23设计模式及其分类,比如 单例模式的8种实现方式、工厂模式的3种实现方式、适配器模式的3种实现、代理模式的3种方式、深拷贝等3) 如果你想写出规范、漂亮的程序,就花时间来学习下设计模式吧课程内容和目标本课程是使用Java来讲解设计模式,考虑到设计模式比较抽象,授课采用 图解+框架源码分析的方式1) 内容包括: 设计模式七大原则(单一职责、接口隔离、依赖倒转、里氏替换、开闭原则、迪米特法则、合成复用)、UML类图(类的依赖、泛化和实现、类的关联、聚合和组合) 23设计模式包括:创建型模式:单例模式(8种实现)、抽象工厂模式原型模式、建造者模式、工厂模式。结构型模式:适配器模式(3种实现)、桥接模式、装饰模式、组合模式、外观模式、享元模式、代理模式(3种实现)。行为型模式:模版方法模式、命令模式、访问者模式、迭代器模式、观察者模式、中介者模式、备忘录模式、解释器模式(Interpreter模式)、状态模式、策略模式、职责链模式(责任链模式)2) 学习目标:通过学习,学员能掌握主流设计模式,规范编程风格,提高优化程序结构和效率的能力。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值