原型模式

原型模式概述

某些特定的时候,我们需要同一个类生成多个相同的对象,或者是基于一个对象生成一个复制对象,并再对复制对象进行修改,这时候就可以使用到原型模式。

UML

一个简单的示例:

package com.prototype;

public class Customer implements Cloneable {
	private String name;
	private int createYear;
	// 由于Address是引用对象, 所以address也需要实现clone方法,防止只实现了浅克隆
	private Address address;

	public String getName() {
		return name;
	}

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

	public int getCreateYear() {
		return createYear;
	}

	public void setCreateYear(int createYear) {
		this.createYear = createYear;
	}

	public Address getAddress() {
		return address;
	}

	public void setAddress(Address address) {
		this.address = address;
	}

	@Override
	protected Object clone() throws CloneNotSupportedException {
		Customer customer = new Customer();
		customer.setCreateYear(this.createYear);
		customer.setName(this.name);
		// 不应该使用这行代码,因为是引用对象,
		// 所以需要深克隆,否则Customerde克隆对象address的修改会影响被克隆对象的address
		// customer.setAddress(this.address);
		customer.setAddress((Address) this.address.clone());
		return customer;
	}

	@Override
	public String toString() {
		return "Customer [name=" + name + ", createYear=" + createYear
				+ ", address=" + address + "]";
	}

}
package com.prototype;

public class Address implements Cloneable {
	private String city;
	private String region;

	public Address(String city, String region) {
		super();
		this.city = city;
		this.region = region;
	}

	@Override
	protected Object clone() throws CloneNotSupportedException {
		Address address = new Address(this.city, this.region);
		return address;
	}

	public String getCity() {
		return city;
	}

	public void setCity(String city) {
		this.city = city;
	}

	public String getRegion() {
		return region;
	}

	public void setRegion(String region) {
		this.region = region;
	}

	@Override
	public String toString() {
		return "Address [city=" + city + ", region=" + region + "]";
	}
}
package com.prototype;

public class Client {
	public static void main(String[] args) throws CloneNotSupportedException {
		Customer customer1 = new Customer();
		customer1.setName("经销商A");
		customer1.setCreateYear(3);
		customer1.setAddress(new Address("Guangzhou", "tianhe"));
		Customer customer2 = (Customer) customer1.clone();
		customer2.setName("经销商B");
		// 大家可以试一下如果Address没有实现Cloneable接口和clone方法的话,会出现什么效果
		customer2.getAddress().setCity("Shenzhen");
		System.out.println(customer1);
		System.out.println(customer2);
	}
}

总结 

使用原型模式可以在不需要知道对象的创建细节的情况下,生成一个与原有对象相同的对象,不易出错且java给我们提供了Cloneable接口可以标记出该类是否支持克隆,但是这里又涉及了浅复制和深复制两种情况,在java中要实现深复制需要在被引用的类中再单独实现克隆方法,可能在这里就会比较繁琐。

才疏学浅,如文中有错误,感谢大家指出。

原文地址:https://www.cnblogs.com/runningRookie/p/11108773.html