- 浏览: 328237 次
- 性别:
- 来自: 杭州
-
文章分类
最新评论
-
arlenliugj:
才发现这贴子好早
如何在Ubuntu下安装windows7 -
arlenliugj:
请问一下,这样安装上windows会不会把已经装好的linux ...
如何在Ubuntu下安装windows7 -
zhaomengbin:
写的很不错,可以写个文件给合并的方法么?,将分割后的几份文件还 ...
文件分割程序 -
junhe0723:
3Q,刚出现这个问题解决了。
jvm terminated exit code 1 -
Anddy:
shell 双击选中太不智能了。
XSHELL快捷键设置
原型模式(Prototype Pattern)
前面的几种模式中,我们使用了不同的构造方法(各种Factory或者Builder)去代替或者说掩盖Java语言之中“new”这个操作来创建对象实例。Java中要创建一个新的对象并不一定只能靠“new”这个关键字的,我们还有“clone()”。
在接触原型模式之前,我们先来了解一下克隆一些知识:
1.clone()方法在Java中从Object类开始就具备,并且作为原生(Native)方法出现。它默认是protected的,因此在被子类提升可见性之前,无法被外界使用。
2.所有需要进行克隆操作的类都必须实现Cloneable接口,这个接口与Serializable接口一样,没有任何需要实现的方法,仅仅是作为一个标示。
3.所有数组都实现了Cloneable接口,并且已经提升了clone()方法的可见性至public,换句话说数组对象是可以直接调用clone()方法的。
4.克隆分为浅拷贝和深拷贝两种,浅拷贝的操作基本上可以理解为只拷贝存储于栈中的内容,包括对象中简单类型的数据、指向其他复杂对象的指针等,但是不会将指向的复杂对象也拷贝一次。
目的:
用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。
场景:
大话西游中唐僧说到:“人是人他妈生的,妖是妖他妈生的,人妖自然是人妖他妈生的咯”,那Zerg众多可爱……可怕的怪兽们都是谁生的?
早上9:00,Kerrigan从床上起来,看见基地里面一片欣欣向荣的景象觉得心情特别舒畅:一群Drone在卖力的挖水晶搬气体,Zergling、Hydralisk们在精神抖擞的踏着正步(BUG:Hydralisk貌似很难走正步),就连基地刚刚分泌出来的3只Larva(虫族的幼虫)都在屁颠屁颠的使劲蹦跶。
Kerrigan也开始了她虫族女王的工作:看了看水晶和气体存量,指着一只Zergling对Larva说:你变成它吧,再指着另外一只Hydralisk对Larva说:你就变成它吧……
分析:
如果说Zergling的存在是为了锋利的抓牙,Hydralisk的存在是为了强腐蚀性的毒液,那Larva生命的唯一意义就是变成别人,它可以说就是Zerg们“它妈”。用一句OOP的话来说:每个Zerg都是从Larva派生出来的,Larva有一个clone方法,自然从Larva派生出来的对象就具备了一个clone方法,它的作用就是根据已有对象(prototype)去生产出一个新的对象。
我们可以用下面代码来模拟“Kerrigan指着一只Zergling对Larva说:你变成它”这个过程:
图5.1 原型模式的UML图
刚刚说了,Larva存在的唯一目的就是为了变成别人,所以它不具备任何的attack()、patrol()方法,只有一个clone()方法,并且是所有Zerg兵种的父类。在这里为了演示需要,给它赋予了3个额外的方法getName()、setName()和toString(),以便区分各个子类。
/**
* Zerg幼虫,是其他Zerg的父类,实现了Closeable接口
*/
public class Larva implements Cloneable {
private String name = "Larva";
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
/**
* 将Clone方法的可见性从protected提升至public
*/
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
public String toString() {
return "兵种名称:" + name + ", 当前对象的内存地址为:" + super.toString();
}
}
接着,我们建立Zergling和Hydralisk两个子类,在本次演示中,我们只希望知道Zerg是怎么繁殖的,而不需要去战斗,所以子类写的非常简单,仅仅赋予它们名字的差别:
/**
* 模拟Hydralisk
*/
public class Hydralisk extends Larva {
public Hydralisk() {
setName("Hydralisk");
}
}
/**
* 模拟Zergling
*/
public class Zergling extends Larva {
public Zergling() {
setName("Zergling");
}
}
到此为止,准备工作还差最后一步就要完成了。要使用原型模式去生产产品,自然就要先有一个原型对象,原型对象怎么来的Prototype不管,是new出来的、由别的工厂建造的、还是实现写在常量中的标本都可以。我个人觉得使用常量标本更加合理一些,请看下面代码:
/**
* 模拟Kerrigan使用原型模式生产Zerg的使用场景
*/
public class Kerrigan {
/**
* 常量:兵种的克隆模版
*/
private static final Larva[] HATCH = new Larva[] { new Hydralisk(), new Zergling() };
/**
* Hydralisk在克隆模版中的序号
*/
private static final int HYDRALISK = 0;
/**
* Zergling在克隆模版中的序号
*/
private static final int ZERGLING = 1;
public static void main(String[] args) throws CloneNotSupportedException {
System.out.println(HATCH[HYDRALISK].clone());
System.out.println(HATCH[HYDRALISK].clone());
System.out.println(HATCH[ZERGLING].clone());
System.out.println(HATCH[ZERGLING].clone());
}
}
在上述代码中我们建立了一个常量模版数组:HATCH,里面有Zergling各种兵种的模版,要生产具体兵种的时候,只需要在模版中选择适合的原型(prototype)进行克隆。这样的具备以下特点:
1.继承对象的属性值:其他的构造模式中,对象具备什么属性值(譬如演示中的Name属性),是由类构造时赋值所决定的,一般来说工厂生产出来的产品属性值都具备一致性。而原型模式生产出来的产品,属性值来源于原型对象而不是类,当前对象的属性值是什么样子,克隆对象的属性值就是什么样子,这点是Prototype模式与其他模式使用场景上最大的差异。
2.运行时修改产品列表:Prototype模式不依赖工厂,只需要拿到实例即可创建新的对象,这点比其他模式更为灵活。
3.绕过对象构造:原型模式生产对象的过程中,本质上讲是一个内存复制的过程,因此它是不会调用对象的构造函数的。
展示一下程序的最终运行结果:
图 5.2 克隆的兵种
总结:
Prototype模式中实现起来最困难的地方就是内存复制操作,所幸在Java中提供了clone()方法替我们做了绝大部分事情。在其他语言之中,一种比较简单的方法是可以考虑使用序列化技术(Serilization)来完成对象的复制。
在Java语言中clone()方法完成的是浅拷贝的克隆,如果需要深拷贝,也可以考虑使用序列化来完成,当然这样比起Native的clone()方法来说,效率差了很多,所以更加推荐的方法是针对类中包含的复杂对象情况,重写clone()方法,多次调用父类的clone()来完成,虽然要多写不少代码,但是保证了效率。
在本章中,我们了解到了Zerg们的出生过程,如果说茫茫多的怪兽都是通过原型模式克隆出来的,那独一无二的虫族刀锋女王Kerrigan呢?请继续看构造模式部分的最后一章——单例模式。
前面的几种模式中,我们使用了不同的构造方法(各种Factory或者Builder)去代替或者说掩盖Java语言之中“new”这个操作来创建对象实例。Java中要创建一个新的对象并不一定只能靠“new”这个关键字的,我们还有“clone()”。
在接触原型模式之前,我们先来了解一下克隆一些知识:
1.clone()方法在Java中从Object类开始就具备,并且作为原生(Native)方法出现。它默认是protected的,因此在被子类提升可见性之前,无法被外界使用。
2.所有需要进行克隆操作的类都必须实现Cloneable接口,这个接口与Serializable接口一样,没有任何需要实现的方法,仅仅是作为一个标示。
3.所有数组都实现了Cloneable接口,并且已经提升了clone()方法的可见性至public,换句话说数组对象是可以直接调用clone()方法的。
4.克隆分为浅拷贝和深拷贝两种,浅拷贝的操作基本上可以理解为只拷贝存储于栈中的内容,包括对象中简单类型的数据、指向其他复杂对象的指针等,但是不会将指向的复杂对象也拷贝一次。
目的:
用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。
场景:
大话西游中唐僧说到:“人是人他妈生的,妖是妖他妈生的,人妖自然是人妖他妈生的咯”,那Zerg众多可爱……可怕的怪兽们都是谁生的?
早上9:00,Kerrigan从床上起来,看见基地里面一片欣欣向荣的景象觉得心情特别舒畅:一群Drone在卖力的挖水晶搬气体,Zergling、Hydralisk们在精神抖擞的踏着正步(BUG:Hydralisk貌似很难走正步),就连基地刚刚分泌出来的3只Larva(虫族的幼虫)都在屁颠屁颠的使劲蹦跶。
Kerrigan也开始了她虫族女王的工作:看了看水晶和气体存量,指着一只Zergling对Larva说:你变成它吧,再指着另外一只Hydralisk对Larva说:你就变成它吧……
分析:
如果说Zergling的存在是为了锋利的抓牙,Hydralisk的存在是为了强腐蚀性的毒液,那Larva生命的唯一意义就是变成别人,它可以说就是Zerg们“它妈”。用一句OOP的话来说:每个Zerg都是从Larva派生出来的,Larva有一个clone方法,自然从Larva派生出来的对象就具备了一个clone方法,它的作用就是根据已有对象(prototype)去生产出一个新的对象。
我们可以用下面代码来模拟“Kerrigan指着一只Zergling对Larva说:你变成它”这个过程:
图5.1 原型模式的UML图
刚刚说了,Larva存在的唯一目的就是为了变成别人,所以它不具备任何的attack()、patrol()方法,只有一个clone()方法,并且是所有Zerg兵种的父类。在这里为了演示需要,给它赋予了3个额外的方法getName()、setName()和toString(),以便区分各个子类。
/**
* Zerg幼虫,是其他Zerg的父类,实现了Closeable接口
*/
public class Larva implements Cloneable {
private String name = "Larva";
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
/**
* 将Clone方法的可见性从protected提升至public
*/
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
public String toString() {
return "兵种名称:" + name + ", 当前对象的内存地址为:" + super.toString();
}
}
接着,我们建立Zergling和Hydralisk两个子类,在本次演示中,我们只希望知道Zerg是怎么繁殖的,而不需要去战斗,所以子类写的非常简单,仅仅赋予它们名字的差别:
/**
* 模拟Hydralisk
*/
public class Hydralisk extends Larva {
public Hydralisk() {
setName("Hydralisk");
}
}
/**
* 模拟Zergling
*/
public class Zergling extends Larva {
public Zergling() {
setName("Zergling");
}
}
到此为止,准备工作还差最后一步就要完成了。要使用原型模式去生产产品,自然就要先有一个原型对象,原型对象怎么来的Prototype不管,是new出来的、由别的工厂建造的、还是实现写在常量中的标本都可以。我个人觉得使用常量标本更加合理一些,请看下面代码:
/**
* 模拟Kerrigan使用原型模式生产Zerg的使用场景
*/
public class Kerrigan {
/**
* 常量:兵种的克隆模版
*/
private static final Larva[] HATCH = new Larva[] { new Hydralisk(), new Zergling() };
/**
* Hydralisk在克隆模版中的序号
*/
private static final int HYDRALISK = 0;
/**
* Zergling在克隆模版中的序号
*/
private static final int ZERGLING = 1;
public static void main(String[] args) throws CloneNotSupportedException {
System.out.println(HATCH[HYDRALISK].clone());
System.out.println(HATCH[HYDRALISK].clone());
System.out.println(HATCH[ZERGLING].clone());
System.out.println(HATCH[ZERGLING].clone());
}
}
在上述代码中我们建立了一个常量模版数组:HATCH,里面有Zergling各种兵种的模版,要生产具体兵种的时候,只需要在模版中选择适合的原型(prototype)进行克隆。这样的具备以下特点:
1.继承对象的属性值:其他的构造模式中,对象具备什么属性值(譬如演示中的Name属性),是由类构造时赋值所决定的,一般来说工厂生产出来的产品属性值都具备一致性。而原型模式生产出来的产品,属性值来源于原型对象而不是类,当前对象的属性值是什么样子,克隆对象的属性值就是什么样子,这点是Prototype模式与其他模式使用场景上最大的差异。
2.运行时修改产品列表:Prototype模式不依赖工厂,只需要拿到实例即可创建新的对象,这点比其他模式更为灵活。
3.绕过对象构造:原型模式生产对象的过程中,本质上讲是一个内存复制的过程,因此它是不会调用对象的构造函数的。
展示一下程序的最终运行结果:
图 5.2 克隆的兵种
总结:
Prototype模式中实现起来最困难的地方就是内存复制操作,所幸在Java中提供了clone()方法替我们做了绝大部分事情。在其他语言之中,一种比较简单的方法是可以考虑使用序列化技术(Serilization)来完成对象的复制。
在Java语言中clone()方法完成的是浅拷贝的克隆,如果需要深拷贝,也可以考虑使用序列化来完成,当然这样比起Native的clone()方法来说,效率差了很多,所以更加推荐的方法是针对类中包含的复杂对象情况,重写clone()方法,多次调用父类的clone()来完成,虽然要多写不少代码,但是保证了效率。
在本章中,我们了解到了Zerg们的出生过程,如果说茫茫多的怪兽都是通过原型模式克隆出来的,那独一无二的虫族刀锋女王Kerrigan呢?请继续看构造模式部分的最后一章——单例模式。
发表评论
-
访问者模式
2011-02-21 17:25 951话说有一个银行,有三个窗口,但是每个窗口的智能都是一样的,即都 ... -
迭代器模式
2011-02-21 17:25 1006【迭代器模式】 迭代器可以顺序访问一个聚集中的元素而不必显露聚 ... -
装饰器模式
2011-02-21 17:25 1985浅谈装饰器模式 序: 今天,为了满足我们项目组长的愿望, ... -
解释器模式
2011-02-21 17:24 1073一、引子 其实没有什么好的例子引入解释器模式,因为它描述了 ... -
命令模式
2011-02-21 17:24 10351.意图: 将一个请求或操作封装到对象中。 2 ... -
策略模式
2011-02-21 17:23 905当我们掌握了Java的语法,当我们了解了面向对象的封装、继承、 ... -
状态模式
2011-02-21 17:23 11461,状态模式允许一个"对象"在其内部状态改 ... -
模板模式
2011-02-21 17:23 1199模板方法(Template Method)模式是GOF设计模式 ... -
备忘录模式 (下)
2011-02-21 17:22 2157一、什么是备忘录模式 Memento模式也叫备忘录模式 ... -
备忘录
2011-02-21 17:22 1109备忘录(Memento Pattern)模式 备忘录模式又叫 ... -
原型模式Prototype(深拷贝)
2011-02-21 17:21 18861、定义:原型模式就是 ... -
抽象工厂模式
2011-02-21 17:20 1378假如你要制作一个对话框控件,你希望这个对话框可以有不同的Loo ... -
工厂方法模式
2011-02-21 17:20 11621、工厂模式的核心思想及分类 工厂方法模式的作用是 ... -
生成器模式
2011-02-21 17:19 1111实际上,既然Builder和Factory同属创建型模式,那么 ... -
构造者模式
2011-02-21 17:19 1164多种设计模式能够解决 ... -
亨元模式
2011-02-21 17:18 1160翻译为亨元模式,或直译为轻量级模式。所谓亨元,就是被其它对象共 ... -
责任链模式
2011-02-21 17:18 6672责任链模式(Chain of Resp ... -
代理模式
2011-02-21 17:17 883代理模式:给某一对象提供代理对象,并由代理对象控制具体对象的引 ... -
观察者模式
2011-02-21 17:17 1041size=9] 论坛上很多人都讲设计模式,也讲了很多设计模式, ... -
多线程下的单例模式(上)
2011-02-21 17:07 1094Abstract 在开发中,如果某个实例的 ...
相关推荐
原型模式是设计模式中的一种,它是Java 23种经典设计模式之一,主要用来提高对象创建的效率。在原型模式中,我们通过复制或克隆一个已经存在的对象来创建新的对象,而不是通过构造函数来创建。这种方法尤其适用于当...
原型模式(Prototype Pattern)是软件设计模式中的一种结构型模式,它的主要目的是通过复制已有对象来创建新对象,从而减少创建新对象的成本。在原型模式中,类的实例化过程被替换为对已有实例的克隆操作,尤其适用...
原型模式(Prototype Pattern)是其中一种行为设计模式,主要用于对象创建。它通过复制已有对象来创建新对象,而不是通过传统的构造函数来创建。在Java中,原型模式可以有效地提高性能,特别是在创建复杂对象时。 #...
本文将深入探讨Android设计模式中的“原型模式”(Prototype Pattern),并结合提供的"prototype"压缩包中的示例代码进行解析。 原型模式是一种创建型设计模式,它的主要思想是通过复制已有对象来创建新对象,而...
原型模式(Prototype Pattern)是一种创建型设计模式,它允许我们通过复制现有的对象来创建新对象,而无需知道具体创建过程的细节。这种模式的核心在于,它提供了一种对象克隆的简便方法,使得对象的创建过程对用户...
原型模式是一种创建型设计模式,它提供了一种创建对象的最佳方式。在原型模式中,一个对象可以被克隆以创建新对象,而无需知道具体的创建细节。这种模式在需要重复创建相似对象时非常有用,避免了每次创建新对象时都...
本篇文章将深入探讨“原型模式(Prototype)”这一经典的设计模式,它是面向对象设计的一个重要概念,尤其在C++编程中有着广泛的应用。 原型模式是一种创建型设计模式,它的核心思想是通过复制已有对象来创建新对象,...
原型模式(Prototype Pattern)是其中一种行为设计模式,它允许我们通过复制现有对象来创建新对象,而不是通过传统方式实例化新对象。这一模式尤其适用于创建复杂或昂贵的对象,因为它提供了对象克隆的能力,从而...
### Java设计模式之原型模式深度解析 #### 模式动机 在面向对象编程的世界里,对象的创建往往伴随着复杂的逻辑处理。特别是在某些场景下,对象的构造过程可能涉及大量资源的消耗,例如数据库连接、文件读写等。在...
java设计模式【之】原型模式、深拷贝与浅拷贝【源码】【场景:克隆羊】 * 原型模式(Prototype) * 实现方式: * 需要被克隆的 class类, 重写Object中的clone()方法,并实现Cloneable接口(否则报错 ...
ava常用设计模式-原型模式 原型模式(Prototype Pattern)是一种创建型设计模式,允许通过复制现有对象来创建新对象,而不是通过实例化类来创建新对象。在需要创建大量相似对象时非常有用,它可以避免重复创建对象...
【设计模式之原型模式】 设计模式是软件工程中的一种最佳实践,是对在特定上下文中反复出现的软件设计问题的解决方案。原型模式属于对象创建型模式,它的主要思想是通过复制已有对象来创建新对象,降低了类的实例化...
设计模式 的分类 总体来说设计模式分为三大类: 创建型模式(5): 工厂方法模式 、抽象工厂模式、单例模式、建造者模式、原型模式。 结构型模式(7): 适配器模式、装饰器模式、代理模式、外观模式、桥接模式、...
C#23种设计模式样例代码和UML图等 创建型模式(抽象工厂模式、工厂方法模式、单例模式、建造者模式、原型模式); 行为型模式(策略模式、 迭代器模式、原型模式、职责链模式、 模板方法、 命令模式、 解释器模式、 ...
JDK中许多类和方法都使用了设计模式,这些模式的应用帮助实现了代码的高内聚、低耦合,提高了代码的可维护性和扩展性。下面介绍几种常见的设计模式及其在JDK中的应用实例: a) 抽象工厂模式(AbstractFactory) ...
本资料集包含了多种重要的前端设计模式,如工厂模式、原型模式和构造器模式,下面将详细阐述这些模式的核心概念和应用。 1. 工厂模式: 工厂模式是一种创建型设计模式,它提供了一种创建对象的最佳方式,通过抽象出...