java学习---对象的克隆和序列化

1、假clone,虚拟机只是将对象的内存地址clone了一份,修改克隆后的对象内容,将影响原对象

package clone;

public class Employee
{
	public Employee()
	{

	}

	public Employee(String name, int age)
	{
		this.age = age;
		this.name = name;
	}

	@Override
	public String toString()
	{
		return "姓名: " + name + "年龄: " + age;
	}

	public String getName()
	{
		return name;
	}

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

	public int getAge()
	{
		return age;
	}

	public void setAge(int age)
	{
		this.age = age;
	}

	public static void main(String[] args)
	{
		Employee demo1 = new Employee("rollen", 20);
		System.out.println(demo1);
		Employee demo2 = demo1;
		demo2.setAge(100);
		demo2.setName("hello world");
		System.out.println(demo1);
		System.out.println(demo2);
	}

	private String name;
	private int age;
}

运行结果:


============================================================================================

2、含有基本数据类型的对象的克隆,必须实现clone方法。克隆后对象属性的修改将不影响原对象的属性

package clone;

class Address
{
	public Address()
	{

	}

	public Address(String state, int number)
	{
		this.number = number;
		this.state = state;
	}

	@Override
	public String toString()
	{
		return "state: " + state + " munber: " + number;
	}

	public String getState()
	{
		return state;
	}

	public void setState(String state)
	{
		this.state = state;
	}

	public int getNumber()
	{
		return number;
	}

	public void setNumber(int number)
	{
		this.number = number;
	}

	private String state;
	private int number;
}

public class Employee2 implements Cloneable
{
	public Employee2()
	{

	}

	public Employee2(String name, int age, Address address)
	{
		this.address = address;
		this.age = age;
		this.name = name;
	}

	public String getName()
	{
		return name;
	}

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

	public int getAge()
	{
		return age;
	}

	public void setAge(int age)
	{
		this.age = age;
	}

	public Address getAddress()
	{
		return address;
	}

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

	@Override
	public String toString()
	{
		StringBuilder sb = new StringBuilder();
		sb.append("name:" + name + ", ");
		sb.append("age:" + age + " \n");
		sb.append("Address: " + address);
		return sb.toString();
	}

	@Override
	protected Employee2 clone()
	{
		Employee2 employee = null;
		try
		{
			employee = (Employee2) super.clone();
		} catch (CloneNotSupportedException e)
		{
			e.printStackTrace();
		}
		return employee;
	}

	public static void main(String[] args)
	{
		System.out.println("克隆之前:");
		Address add1 = new Address("中国", 1);
		Employee2 emp1 = new Employee2("rollen", 20, add1);
		System.out.println(emp1);
		System.out.println("克隆之后");
		Employee2 emp2 = emp1.clone();
		emp2.setName("hello world");
		emp2.setAge(100);
		emp2.address.setNumber(2);
		emp2.address.setState("美国");
		System.out.println(emp1);
		System.out.println("-----");
		System.out.println(emp2);
	}

	private String name;
	private int age;
	private Address address;
}

【运行结果】:


==============================================================================================

3、含有复制数据结构类的类的克隆,对类中复杂的数据类型也必须实现clone方法。

package clone;

class Address implements Cloneable
{
	public Address()
	{

	}

	public Address(String state, int number)
	{
		this.number = number;
		this.state = state;
	}

	@Override
	public String toString()
	{
		return "state: " + state + " munber: " + number;
	}

	@Override
	protected Address clone() throws CloneNotSupportedException
	{
		Address address = null;
		address = (Address) super.clone();
		return address;
	}

	public String getState()
	{
		return state;
	}

	public void setState(String state)
	{
		this.state = state;
	}

	public int getNumber()
	{
		return number;
	}

	public void setNumber(int number)
	{
		this.number = number;
	}

	private String state;
	private int number;
}

public class Employee3 implements Cloneable
{
	public Employee3()
	{

	}

	public Employee3(String name, int age, Address address)
	{
		this.address = address;
		this.age = age;
		this.name = name;
	}

	public String getName()
	{
		return name;
	}

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

	public int getAge()
	{
		return age;
	}

	public void setAge(int age)
	{
		this.age = age;
	}

	public Address getAddress()
	{
		return address;
	}

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

	@Override
	public String toString()
	{
		StringBuilder sb = new StringBuilder();
		sb.append("name:" + name + ", ");
		sb.append("age:" + age + " \n");
		sb.append("Address: " + address);
		return sb.toString();
	}

	@Override
	protected Employee3 clone()
	{
		Employee3 employee = null;
		try
		{
			employee = (Employee3) super.clone();
			employee.address = address.clone();
		} catch (CloneNotSupportedException e)
		{
			e.printStackTrace();
		}
		return employee;
	}

	public static void main(String[] args)
	{
		System.out.println("克隆之前:");
		Address add1 = new Address("中国", 1);
		Employee3 emp1 = new Employee3("rollen", 20, add1);
		System.out.println(emp1);
		System.out.println("克隆之后");
		Employee3 emp2 = emp1.clone();
		emp2.setName("hello world");
		emp2.setAge(100);
		emp2.setAddress(new Address("美国", 2));
		System.out.println(emp1);
		System.out.println("-----");
		System.out.println(emp2);
	}

	private String name;
	private int age;
	private Address address;
}

【运行结果】:


=========================================================================================

4、序列化接口和对象克隆

Java 串行化技术可以使你将一个对象的状态写入一个Byte 里,并且可以从其它地方把该Byte 流里的数据读出来,重新构造一个相同的对象。这种机制允许你将对象通过网络进行传播,并可以随时把对象持久化到数据库、文件等系统里。Java的串行化机制是RMI、EJB等技术的技术基础。用途:利用对象的串行化实现保存应用程序的当前工作状态,下次再启动的时候将自动地恢复到上次执行的状态。

序列化就是一种用来处理对象流的机制,所谓对象流也就是将对象的内容进行流化。可以对流化后的对象进行读写操作,也可将流化后的对象传输于网络之间。序列化是为了解决在对对象流进行读写操作时所引发的问题。

序列化的实现:将需要被序列化的类实现Serializable接口,然后使用一个输出流(如:FileOutputStream)来构造一个ObjectOutputStream(对象流)对象,接着,使用ObjectOutputStream对象的writeObject(Object obj)方法就可以将参数为obj的对象写出(即保存其状态),要恢复的话则用输入流。

2、串行化的特点:

    (1)如果某个类能够被串行化,其子类也可以被串行化。如果该类有父类,则分两种情况来考虑,如果该父类已经实现了可串行化接口。则其父类的相应字段及属性的处理和该类相同;如果该类的父类没有实现可串行化接口,则该类的父类所有的字段属性将不会串行化。

  (2)声明为statictransient类型的成员数据不能被串行化。因为static代表类的状态, transient代表对象的临时数据;

  (3)相关的类和接口:在java.io包中提供的涉及对象的串行化的类与接口有ObjectOutput接口、ObjectOutputStream类、ObjectInput接口、ObjectInputStream类。

    (1)ObjectOutput接口:它继承DataOutput接口并且支持对象的串行化,其内的writeObject()方法实现存储一个对象。ObjectInput接口:它继承DataInput接口并且支持对象的串行化,其内的readObject()方法实现读取一个对象。

    (2)ObjectOutputStream类:它继承OutputStream并且实现ObjectOutput接口。利用该类来实现将对象存储(调用ObjectOutput接口中的writeObject()方法)。ObjectInputStream类:它继承InputStream类并且实现ObjectInput接口。利用该类来实现读取一个对象(调用ObjectInput接口中的readObject()方法)。

  对于父类的处理,如果父类没有实现串行化接口,则其必须有默认的构造函数(即没有参数的构造函数)。否则编译的时候就会报错。在反串行化的时候,默认构造函数被调用。但是若把父类标记为可以串行化,则在反串行化的时候,其默认构造函数不会被调用。这是为什么呢?这是因为Java 对串行化的对象进行反串行化的时候,直接从流里获取其对象数据来生成一个对象实例,而不是通过其构造函数来完成。

import java.io.*;
public class Cat implements Serializable {
        private String name;
        public Cat () {
                this.name = "new cat";
        }
        public String getName() {
                return this.name;
        }
        public void setName(String name) {
                this.name = name;
        }
        public static void main(String[] args) {         
                Cat cat = new Cat();
                try {
                        FileOutputStream fos = new FileOutputStream("catDemo.out");
                        ObjectOutputStream oos = new ObjectOutputStream(fos);
                        System.out.println(" 1> " + cat.getName());
                        cat.setName("My Cat");                       
                        oos.writeObject(cat);
                        oos.close();                       
                } catch (Exception ex) {  ex.printStackTrace();   }
                try {
                        FileInputStream fis = new FileInputStream("catDemo.out");
                        ObjectInputStream ois = new ObjectInputStream(fis);
                        cat = (Cat) ois.readObject();
                        System.out.println(" 2> " + cat.getName());
                        ois.close();
                } catch (Exception ex) {
                        ex.printStackTrace();
                }
        }
}

//writeObject和readObject本身就是线程安全的,传输过程中是不允许被并发访问的。所以对象能一个一个接连不断的传过来


<pre name="code" class="java">package clone;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

class Address implements Serializable
{
	public Address()
	{

	}

	public Address(String state, int number)
	{
		this.number = number;
		this.state = state;
	}

	@Override
	public String toString()
	{
		StringBuilder sb = new StringBuilder();
		sb.append("number: " + number);
		sb.append("state" + state + "\n");
		return sb.toString();
	}

	public String getState()
	{
		return state;
	}

	public void setState(String state)
	{
		this.state = state;
	}

	public int getNumber()
	{
		return number;
	}

	public void setNumber(int number)
	{
		this.number = number;
	}

	private String state;
	private int number;
}

public class Employee implements Cloneable, Serializable
{
	public Employee()
	{

	}

	public Employee(String name, int age, Address address)
	{
		this.address = address;
		this.age = age;
		this.name = name;
	}

	public String getName()
	{
		return name;
	}

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

	public int getAge()
	{
		return age;
	}

	public void setAge(int age)
	{
		this.age = age;
	}

	public Address getAddress()
	{
		return address;
	}

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

	@Override
	public String toString()
	{
		StringBuilder sb = new StringBuilder();
		sb.append("name:" + name + ", ");
		sb.append("age:" + age + " \n");
		sb.append("Address: " + address);
		return sb.toString();
	}

	@Override
	protected Employee clone()
	{
		Employee employee = null;
		ByteArrayOutputStream baos = new ByteArrayOutputStream();
		try
		{
			ObjectOutputStream oos = new ObjectOutputStream(baos);
			oos.writeObject(this);
			oos.close();

			ByteArrayInputStream bais = new ByteArrayInputStream(
					baos.toByteArray());
			ObjectInputStream ois = new ObjectInputStream(bais);
			employee = (Employee) ois.readObject();
			ois.close();
		} catch (Exception e)
		{
			e.printStackTrace();
		}
		return employee;
	}

	public static void main(String[] args)
	{
		System.out.println("克隆之前:");
		Address add1 = new Address("中国", 1);
		Employee emp1 = new Employee("rollen", 20, add1);
		System.out.println(emp1);
		System.out.println("克隆之后");
		Employee emp2 = emp1.clone();

		emp2.setName("hello world");
		emp2.setAge(100);
		emp2.address.setNumber(2);
		emp2.address.setState("美国");
		System.out.println(emp1);
		System.out.println("-----");
		System.out.println(emp2);
	}

	private String name;
	private int age;
	private Address address;
}


 

克隆之前:





评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值