原型模式

使用原型模式来复制一个对象,从而克隆出多个与原型对象一模一样的对象。

在某些系统中,有些对象的创建过程很昂贵很复杂,而且有时候需要频繁创建。

可以用原型模式解决。

 

原型模式(Prototype Pattern):原型模式是一种对象创建型模式,用原型实例,指定创建对象的种类,并且通过复制这些原型,创建新的对象。

 

在原型模式结构中定义了一个抽象原型类,所有的Java类都继承自java.lang.Object,而Object类提供一个clone()方法,可以将一个Java对象复制一份。因此在Java中可以直接使用Object提供的clone()方法来实现对象的克隆(浅拷贝),Java语言中的原型模式实现很简单。

能够实现克隆的Java类必须实现一个标识接口Cloneable,表示这个Java类支持复制。如果一个类没有实现这个接口但是调用了clone()方法,Java编译器将抛出一个CloneNotSupportedException异常。 

 

浅拷贝是指拷贝对象时仅仅拷贝对象本身(包括对象中的基本变量,引用变量地址),而不拷贝对象包含的引用指向的对象。

深拷贝不仅拷贝对象本身,而且拷贝对象包含的引用指向的所有对象。

 

若不对clone()方法进行改写,则调用此方法得到的对象即为浅拷贝(Object的clone为浅拷贝)。

深拷贝方法,就是将对象串行化。

 

package com.ez;

import java.io.Serializable;

/**
 * 浅拷贝,通过Object提供的clone方法,就可以实现浅拷贝。
 * 对象必须继承Cloneable接口。
 * @author 窗外赏雪(EZ编程网)
 */
public class People implements Cloneable,Serializable{
	private static final long serialVersionUID = -4084944705840672303L;
	private int age;
	private String name;
	
	public People(int age,String name) {
		this.age=age;
		this.name=name;
	}
	
	public int getAge() {
		return age;
	}
	public String getName() {
		return name;
	}
	public void setAge(int age) {
		this.age = age;
	}
	public void setName(String name) {
		this.name = name;
	}
	/**
	 * 子类默认实现方式:覆盖重写父类的方法,默认使用super.xxx
	 * 所以在子类中,可以调用父类的方法。
	 */
	@Override
	protected Object clone() throws CloneNotSupportedException {
		return super.clone();
	}
	
	public static void main(String[] args) {
		People p0=new People(15,"李四");
		System.out.println("age:"+p0.getAge()+",name:"+p0.getName());;
		try {
			People p1=(People)p0.clone();
			System.out.println("age:"+p1.getAge()+",name:"+p1.getName());
		} catch (CloneNotSupportedException e) {
			e.printStackTrace();
		}
	}
}

 

package com.ez;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OptionalDataException;
import java.io.Serializable;
/**
 * 深拷贝和浅拷贝的区别
 * @author 窗外赏雪(EZ编程网)
 */
public class Student implements Serializable,Cloneable{

	/**
	 * 反序列化的时候,需要对照这个ID,如果不匹配就反序列化异常。
	 */
	private static final long serialVersionUID = 3457679362463213165L;

	private String name;// 常量对象。
    private int age;
    private People people;// 学生1和学生2的引用值都是一样的。
 
    Student(String name, int age, People p) {
        this.name = name;
        this.age = age;
        this.people = p;
    }
 
    public String getName() {
		return name;
	}
	public int getAge() {
		return age;
	}
	public People getPeople() {
		return people;
	}

	public Object deepClone() throws IOException, OptionalDataException,
            ClassNotFoundException {
        // 将对象写到流里
        ByteArrayOutputStream bo = new ByteArrayOutputStream();
        ObjectOutputStream oo = new ObjectOutputStream(bo);
        oo.writeObject(this);
        // 从流里读出来
        ByteArrayInputStream bi = new ByteArrayInputStream(bo.toByteArray());
        ObjectInputStream oi = new ObjectInputStream(bi);
        return (oi.readObject());
    }
    
    public static void main(String[] args) throws CloneNotSupportedException, OptionalDataException, ClassNotFoundException, IOException {
		People p0=new People(18, "窗外赏雪");
		Student stu=new Student("李四", 15, p0);
		//浅拷贝只拷贝p0的引用
		Student stu1=(Student)stu.clone();
		System.out.println("age:"+stu1.getAge()+",name:"+stu1.getName());
		stu1.getPeople().setAge(111);
		stu1.getPeople().setName("大刀关羽");
		System.out.println("======"+p0.getAge()+p0.getName()+"======");
		
		//深拷贝会拷贝p0的真实对象
		Student stu2=(Student)stu.deepClone();
		System.out.println("age:"+stu2.getAge()+",name:"+stu2.getName());
		stu2.getPeople().setAge(222);
		stu2.getPeople().setName("小刀关胜");
		System.out.println("======"+p0.getAge()+p0.getName()+"======");
	}
}

 

原型模式的优点

当创建新的对象实例较为复杂时,使用原型模式可以简化对象的创建过程。(通过一个已有实例可以提高新实例的创建效率)

 

原型模式应用:

Struts2中为了保证线程的安全性,Action对象的创建使用了原型模式,访问一个已经存在的Action对象时将通过克隆的方式创建出一个新的对象,从而保证其中定义的变量无须进行加锁实现同步,每一个Action中都有自己的成员变量,避免Struts1因使用单例模式而导致的并发和同步问题。

 

Spring中,用户也可以采用原型模式来创建新的bean实例,从而实现每次获取的是通过克隆生成的新实例,对其进行修改时对原有实例对象不造成任何影响。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值