今天写程序的时候像直接调用Object的clone()方法来copy一个副本,但是发现myEclipse的提示中并没有该方法,以为在jdk1.7中取消了该方法,然后我直接敲上clone()后:
public class TestObject {
public static void main(String[] args) {
Student s = new Student(1, "小时");
s.clone();//报错
}
}
编译报错:The method clone() from the type Object is not visible
后来看了看源码,发现原来是这样:
Object源码中对于clone()方法的描述:
protected native Object clone() throws CloneNotSupportedException;
native修饰词解析:native的方法就是一个java调用非java代码的接口,是一个其他语言实现的方法。
以前对protected的理解是错误的,protected的方法在自己的包中与public相同
在不同包,指的是通过自身实例(自身的引用)访问,而不能通过父类实例(引用)访问
所以上端代码中调用Student类的clone()方法会包编译错误,下面的代码(通过自身引用调)就OK的
public class TestObject {
public static void main(String[] args) {
TestBoject t = new TestObject();
t.clone();//ok
}
}
问题2:要想实现类似文章开头的代码功能该怎么办?
在Object的源码中注释到:
首先,如果这个类没有实现Cloneable这个接口,将会抛出CloneNotSupportedException,但是所有的数组都被看成实现了这个接口。此方法执行的是该对象的“浅表复制”,而不“深层复制”操作。
Object 类本身不实现接口 Cloneable,所以在类为 Object 的对象上调用 clone 方法将会导致在运行时抛出异常。
在Cloneable的源码中注释到:
此类实现了 Cloneable 接口,以指示 Object.clone() 方法可以合法地对该类实例进行按字段复制。 如果在没有实现 Cloneable 接口的实例上调用 Object 的 clone 方法,则会导致抛出 CloneNotSupportedException异常。
按照惯例,实现此接口的类应该使用公共方法重写 Object.clone(它是受保护的)。请参阅 Object.clone(),以获得有关重写此方法的详细信息。
注意,此接口不包含 clone 方法。因此,因为某个对象实现了此接口就克隆它是不可能的。即使 clone 方法是反射性调用的,也无法保证它将获得成功解读:Cloneable接口没有任何方法,仅是个标志接口(tagging interface),若要具有克隆能力,实现Cloneable接口的类必须重写从Object继承来的clone方法,并调用Object的clone方法(见下面Object#clone的定义),重写后的方法应为public 的。For example(标准写法):
示例代码:
public class TestObject {
public static void main(String[] args) {
Student s = new Student(1, "小时");
System.out.println(s.clone().getClass());
System.out.println(s.clone().equals(s));
}
}
class Student implements Cloneable{
public int id;
public String name;
Student(int id, String name){
this.id = id;
this.name = name;
}
public Object clone(){
Student s = null;
try{
s = (Student)super.clone();
}catch(CloneNotSupportedException e){
e.printStackTrace();
}
return s;
}
}