java clone方法_java基础之Object类解析(一)

fdbe48785608094409088d55b31f554f.png

1. 导读

java中每新增一个类, 他的父类都是Object类, 那么Object类中的方法都会被子类继承, 今天主要来看看Object中的5个方法;

1.1 getClass();

1.2 hashCode();

1.3 equals();

1.4 clone();

1.5 toString();

2. getClass方法

getClass()方法是用来获取运行时的对象; 当声明对象和.class文件中真正的对象不一致时, 该方法会返回.class中的对象;

该方法主要用于JAVA的反射机制;

public final native Class<?> getClass();

从上面的代码中我们需要关注几个关键字:

2.1 final: 说明getClass()是不能被子类重写;

不能被重写是为了保证一个子类有多重继承关系时, 其调用getClass()方法与其父类调用getClass()方法的表现是一致的, 这也是实现JAVA反射的保证;

举个例子, A extends B; 如果B重写了getClass()方法, 返回的Class是B; 当A调用时, A没有重写getClass(), 返回也会是B, 而不是真正的实例A; 这明显与getClass()方法的预期不符, 也会造成使用JAVA反射获取实例时, 获取到的是B实例而不是A实例;

为了防止这种很差的实现, 所以getClass()声明成了final;

2.2 native: java中带有native关键字的方法都是原生方法, 是由JVM底层的C来实现, 这种方式称为JNI(java native interface);

需要注意一点因为JVM并不只有HotSpot, 所以native方法在不同JVM上的表现结果有可能是不一致的;

2.3 Class<?>: getClass()的返回值是Class类型的, Class这个类我们放到后面的反射中再讲解;

划重点:

.1 返回的是运行时的实例类型, 而不是编译类型;

.2 该方法是final的, 子类调用和父类调用的都是Object类的实现;

3. hashCode方法

hashCode()的作用返回一个该对象int类型的哈希码;

public native int hashCode();

我们需要关注的关键字是:

3.1 native: hashCode也是一个原生方法, 那么他的实现依赖于底层的JVM实现;

3.2 final: hashCode没有声明成final, 证明可以被子类重写, 但是重写时需遵从以下三点约定:

.1 在java程序执行过程中,在一个对象没有被改变的前提下,无论这个对象被调用多少次,hashCode方法都会返回相同的int值。这个int值可在不同的程序中不

同;

.2 如果2个对象使用equals方法进行比较并且相同的话,那么这2个对象的hashCode方法的值也必须相等;

.3 如果根据equals方法,得到两个对象不相等,那么这2个对象的hashCode值可以不相同; 但是,不相等的对象的hashCode值不同的话可以提高哈希表的性能;

hashCode重写时为什么需要遵从这三个约定, 放到后面的equals方法中说明;

划重点:

.1 一个相同的类在不同的JVM上调用hashCode可能返回不同的int值;

.2 重写hashCode时需用遵从上面三个约定;

4. equals方法

equals()的作用是比较两个对象是否相等; 默认是实现是比较引用是否相同, 即是否为同一个内存对象;

public boolean equals(Object obj) {

return (this == obj);

}

可以看到equals方法并不是native方法, 已经有了java的默认实现, 并且该方法也没有用final进行修饰, 那么equals和hashCode一样, 是可以被重写的, 那么我们重写时, 是否也需要和hashCode一样遵从某些约定呢? 答案是: 是的;

我们需要遵从下面的约定(基于非空对象的equals调用需满足的特性):

4.1 自反性: 对于任何一个非空对象x, x.equals(x)的结果必定为true;

4.2 对称性: 对于非空对象x 和 y, x.equals(y)的结果与y.equals(x)的结果相同;

4.3 传递性: 对于非空对象x, y 和 z, 如果x.equals(y) == true, y.equals(z) == true, 那么 x.equals(z)的结果必定是true;

4.4 一致性: 对于非空对象x 和 y, 在x 和 y都没做修改的前提下, 多次调用x.equals(y)的结果应是一致的;

4.5 对于非空对象x, x.equals(null)的结果必定为false;

4.6 如果重写了equals方法, 最好重写hashCode; 原因是如果有两个对象A(key:1, value:1) 和 B(key:1, value:1), equals方法重写后根据key,value是否相等来判断, 但是在内存中是两个不同的对象, 那么hashCode的结果有可能是不同的, 也就违反了上面hashCode约定中的第二条;

划重点:

.1 重写equals需要重写hashCode;

5. clone方法

protected native Object clone() throws CloneNotSupportedException;

clone方法创建并返回当前对象的一份拷贝; 一般情况下, 对于任何对象 x, 表达式 x.clone() != x 为true,x.clone().getClass() == x.getClass() 也为true;

Object类的clone方法是一个protected的native方法; 而且Object本身没有实现Cloneable接口, 所以不重写clone方法并且进行调用的话会发生CloneNotSupportedException异常;

clone方法可引申出深拷贝和浅拷贝;

5.1 浅克隆:

97abca36521013fa2d68f095770830bc.png

5.2 深克隆:

eabb2175953338f960b5749f13cb620a.png

从上面深浅克隆的示意图可知: 深浅克隆的区别就是对于克隆对象中的非静态引用类型的处理: 浅克隆不会新增引用对象, 而深克隆则会连引用对象都会克隆一份;

但是如果一个对象只有基本类型, 那么该对象的克隆则没有深浅之分;

至于深拷贝和浅拷贝的详细说明以及如何实现深拷贝, 放到后面再讲;

划重点:

.1 clone方法是可被子类继承的, 但是直接调用会抛异常;

Object类中的其他方法, 将在下一期分享中做解析; 本期中如有错误, 欢迎指正;

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值