8

我正在研究原型设计模式并且有一些问题。

我已经了解原型设计模式用于创建在内存或资源方面成本高昂的对象。在这种情况下,我们使用已经可用的对象的克隆。

new那么创建对象和创建对象有什么区别clone()?对象存储在内存中的什么位置?

4

3 回答 3

5

Javaclone()方法只是创建一个新对象并将成员变量值复制到其中。一般来说,它既不比创建一个新对象的成本高也不低。唯一clone()比创建对象便宜的时候是对象new的构造函数做一些昂贵的事情:例如,如果构造函数接受参数并将它们用作数据库查询的一部分怎么办?在这种情况下,使用clone()会更便宜,因为不会发生昂贵的查询操作。

但是,使用这种设计模式还有其他原因:大多数情况下,对象在使用前需要复杂的设置,而这在构造函数中无法方便地完成。想象一个对象有 20 个需要设置的属性。如果你用构造函数参数来设置它们,那个构造函数会非常难看——想象一个有 20 个参数的构造函数!相反,您可以构造一个可能没有参数的对象,使用 mutator 方法设置 20 个值,然后clone()在需要时将该对象制作成现成的副本。clone()根本不需要参数,所以它显然不那么难看。如果您需要该对象的多个不同版本的多个副本,那么原型模式就会变得很有吸引力。

于 2012-07-29T04:05:16.517 回答
4

原型设计模式提供两种成本节省——时间节省和空间节省。

当创建对象需要对辅助信息进行昂贵的访问时(例如,从文件、数据库或通过网络请求配置数据),可以节省时间。例如,如果您要从存储在 Web 服务器上的模板构建大量页面,则只需读取一次模板并克隆它以获得每个新页面的起点,而不是单独查询 Web 服务器每一页。

内存节省来自重用不可变对象:如果您的原始对象包含大量字符串,则创建一个新实例需要创建全新的不可变字符串,或者手动处理字符串实习。通过让克隆共享模板的不可变部分,使用原型模式优雅地避免了这个问题。

于 2012-07-29T04:10:00.933 回答
0

当对象的创建占用过多的系统资源和性能时使用原型设计模式,而我们正是在我们想要一个类的许多实例时使用这种设计模式,并且这些实例是相似的,所以我们真的不想要例如使用运算符“new”,因为它的成本很高,我们所需要的只是根据已经创建的第一个对象来实例化这些对象。

优点是新对象将是独立的,并且不会像第一个那样创建太多资源。这里是在 java 中使用这个概念的一个例子:

  import java.util.Vector;

 public class Samsung implements Cloneable{
    private Vector<String> models; 

    public Samsung(){
        models=new Vector<>(); 
        //we suppose in this comments we access to a data Base to get models
        //and then we get a full list of Samsung models
        //... and finish
        //Sadly we took to much of time to fetch the database 
        //we don't want to waste our time again because Samsung rarely update its database
        models.add("Samsung S1"); 
        models.add("Samsung S2"); 
        models.add("galaxy note"); 
        models.add("galaxy star");
    }
    public Samsung(Vector<String> models){
        this.models=models;
    }

   public Samsung clone()  {

      Vector<String> modelsCopy=new Vector<>();
      Samsung samsungCopy=null;
    //here we don't need to access the database again, we will just copy the previous list
      try{
          for(String model:this.models){
              modelsCopy.add(model);
          }
          samsungCopy=new Samsung(modelsCopy); 
          return samsungCopy;
      }
      catch(Exception e){
          return null;
      }
}

}

主程序:

  public static void main(String[] args) {
        Samsung usa_Samsung=new Samsung();
        Samsung morocco_Samsung=usa_Samsung.clone(); 
        System.out.println("original = " + usa_Samsung);
        System.out.println("copy = " + morocco_Samsung);
    }

输出 :

original = Samsung@6d06d69c

copy = Samsung@7852e922

就像您看到这些对象的地址不同,因为它们不同。

笔记 !我仅使用名称“三星”作为示例。

于 2018-04-13T23:43:45.763 回答