设计模式--原型模式

原型模式:

       通过new产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式。原型模式就是java中的克隆技术,以某个对象为原型,复制出新的对象。显然,新的对象具备原型对象的特点。

     原型模式优势:效率高(直接克隆,避免了重新执行构造过程步骤)。

     克隆类似于new,但是不同于new。new创建新的对象属性采用的是默认值。克隆出的对象的属性完全和原型对象相同。并且克隆出的新对象改变不会影响原型对象。然后,再修改克隆对象的值。

原型模式实现:

     Cloneable接口和clone方法。(Prototype模式实现起来最困难的地方就是内存复制操作,所幸在java中提供了clone方法替代我们做了绝大部分事情)

/**
 * 猴子
 */
public class Monkey implements Cloneable{
	private String name;
	private Date birthday;
	@Override
	protected Object clone() throws CloneNotSupportedException {
		Object obj = super.clone();//浅复制,直接调用object对象的clone方法,没有调用

Date对象的clone方法
		return obj;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public Date getBirthday() {
		return birthday;
	}
	public void setBirthday(Date birthday) {
		this.birthday = birthday;
	}
	public Monkey(String name, Date birthday) {
		super();
		this.name = name;
		this.birthday = birthday;
	}
	public Monkey() {
	}
}

/**
 *测试原型模式(浅复制)
 */
public class Client {
	public static void main(String[] args) throws CloneNotSupportedException{
		Date date = new Date(500000L);
		Monkey dasheng = new Monkey("孙悟空",date);

		Monkey monkey = (Monkey)dasheng.clone();//克隆出一个新猴子
		
		System.out.println("1.名字:"+dasheng.getName()+"\t\t出生日期:"+dasheng.getBirthday());
		
		System.out.println("------------------修改出生日期--------");
		date.setTime(1234567890L);
		System.out.println("2.名字:"+dasheng.getName()+"\t\t出生日期:"+dasheng.getBirthday());
		
		monkey.setName("孙影---孙悟空的影子");
		System.out.println("3.名字:"+monkey.getName()+"\t\t出生日期:"+monkey.getBirthday());
	}
}
     运行的结果如下:


      由上面运行结果,可知,当修改出生日期date的值,孙悟空和孙影的出生日期的值都跟着改变了,这说明dasheng和monkey对象共同指向了同一个date对象,这就属于浅复制。

      深复制就是当克隆时,把属性对象都重新分配属性对象,也就说克隆前对象和克隆后对象的属性分别指向了不同的地址空间。深复制实现如下:

 

/**
 * 猴子
 */
public class Monkey implements Cloneable{
	private String name;
	private Date birthday;
	@Override
	protected Object clone() throws CloneNotSupportedException {
		Object obj = super.clone();//直接调用object对象的clone方法
		//添加如下代码实现深复制
		Monkey monkey = (Monkey)obj;
		monkey.birthday = (Date)this.birthday.clone();//把Date属性也进行复制
		return monkey;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public Date getBirthday() {
		return birthday;
	}
	public void setBirthday(Date birthday) {
		this.birthday = birthday;
	}
	public Monkey(String name, Date birthday) {
		super();
		this.name = name;
		this.birthday = birthday;
	}
	public Monkey() {
	}
}


/**
 *测试原型模式(深复制)
 */
public class Client {
	public static void main(String[] args) throws CloneNotSupportedException{
		Date date = new Date(500000L);
		Monkey dasheng = new Monkey("孙悟空",date);

		Monkey monkey = (Monkey)dasheng.clone();//克隆出一个新猴子
		
		System.out.println("1.名字:"+dasheng.getName()+"\t\t出生日

期:"+dasheng.getBirthday());
		
		System.out.println("------------------修改出生日期--------");
		date.setTime(1234567890L);
		System.out.println("2.名字:"+dasheng.getName()+"\t\t出生日

期:"+dasheng.getBirthday());
		
		monkey.setName("孙影---孙悟空的影子");
		System.out.println("3.名字:"+monkey.getName()+"\t\t出生日

期:"+monkey.getBirthday());
	}
}
      运行结果如下:


     原型模式的注意事项:
1.使用原型模式复制对象不会调用类的构造方法。因为对象的复制是通过调用Object类的clone方法来完成的,它直接在内存中复制数据,因此不会调用到类的构造方法。不但构造方法中的代码不会执行,甚至连访问权限都对原型模式无效。
2.深拷贝与浅拷贝。Object类的clone方法只会拷贝对象中的基本的数据类型,对于数组、容器对象、引用对象等都不会拷贝,这就是浅拷贝。如果要实现深拷贝,必须将原型模式中的数组、容器对象、引用对象等另行拷贝。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值