您当前的位置: 首页 > 

庄小焱

暂无认证

  • 2浏览

    0关注

    800博文

    0收益

  • 0浏览

    0点赞

    0打赏

    0留言

私信
关注
热门博文

JDK源码——Object类

庄小焱 发布时间:2021-04-29 20:00:53 ,浏览量:2

摘要

本博文主要介绍Object类的源码分析。帮助大家学习和了解Object类。

一、clone() 方法
 protected native Object clone() throws CloneNotSupportedException;

保护方法,实现对象的浅复制,只有实现了Cloneable接口才可以调用该方法,否则抛出CloneNotSupportedException异常。主要是JAVA里除了8种基本类型传参数是值传递,其他的类对象传参数都是引用传递,我们有时候不希望在方法里讲参数改变,这是就需要在类中复写clone方法(实现深复制)。创建并返回此对象的一个副本。“副本”的准确含义可能依赖于对象的类。

1.1 clone与copy的区别

假设现在有一个Employee对象,Employee tobby =new Employee(“CMTobby”,5000), 通常我们会有这样的赋值Employee cindyelf=tobby,这个时候只是简单了copy了一下reference,cindyelf和tobby都指向内存中同一个object,这样cindyelf或者tobby的一个操作都可能影响到对方。打个比方,如果我们通过cindyelf.raiseSalary()方法改变了salary域的值,那么tobby通过,getSalary()方法得到的就是修改之后的salary域的值,显然这不是我们愿意看到的。我们希望得到tobby的一个精确拷贝,同时两者互不影响,这时候   我们就可以使用Clone来满足我们的需求。Employee cindy=tobby.clone(),这时会生成一个新的Employee对象,并且tobby具有相同的属性值和方法。

1.2 深拷贝与浅拷贝的区别

浅拷贝:创建一个新对象,然后将当前对象的非静态字段复制到该新对象,如果字段是值类型的,那么对该字段执行复制;如果该字段是引用类型的话,则复制引用但不复制引用的对象。因此,原始对象及其副本引用同一个对象。如果更改其中的一个基本类型,那么另一个并不会改变。

深拷贝:创建一个新对象,然后将当前对象的非静态字段复制到该新对象,无论该字段是值类型还是引用类型,都复制独立的一份。当你修改其中一个对象的任何内容时,都不会影响另一个对象的内容。

如何实现深拷贝。让每个引用类型属性内部都重写clone()方法,既然引用类型不能实现深拷贝,那么我们将每个引用类型都拆分为基本类型,分别进行浅拷贝。比如上面的例子,Person类有一个引用类型Address(其实String也是引用类型,但是String类型有点特殊),我们在Address类内部也重写clone方法。如下:

class Address implements Cloneable{
    private String provices;
    private String city;
    public void setAddress(String provices,String city){
        this.provices = provices;
        this.city = city;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }

    @Override
    public String toString() {
        return "Address [provices=" + provices + ", city=" + city + "]";
    }
}
protected Object clone() throws CloneNotSupportedException{
    Person p = (Person) super.clone();
    p.address = (Address) address.clone();
    return p;
}

这种做法有种弊端,这里我们Person类只有一个Address引用类型,而Address类没有,所以我们只用重写Address类的clone方法,但是如果Address类也存在一个引用类型,那么我们也要重写其clone方法,这样下去,有多少个引用类型,我们就要重写多少次,如果存在很多引用类型,那么代码量显然会很大。

利用序列化实现深拷贝。序列化是将对象写到流中便于传输,而反序列化则是把对象从流中读取出来。这里写到流中的对象则是原始对象的一个拷贝,因为原始对象还存在JVM中,所以我们可以利用对象的序列化产生克隆对象,然后通过反序列化获取这个对象。注意:每个需要序列化的类都要事先Serializable接口,如果有某个属性不需要序列化,可以将其声明为transient,即将其排除在克隆属性之外。

public Object deepClone() throws Exception{
    //序列化
    ByteArrayOutputStream bos = new ByteArrayOutputStream();
    ObjectOutputStream oos = new ObjectOutputStream(bos);

    oos.write(this);

    //反序列化
    ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
    ObjectInputStream ois = new ObjectInputStream(bis);
    
    return ois.readObject();
}

序列化产生的是两个完全独立的对象,所以无论嵌套多少个引用类型,序列化都能实现深拷贝的。

二、tostring()方法

Object 类的 toString 方法返回一个字符串,该字符串由类名(对象是该类的一个实例)、at 标记符“@”和此对象哈希码的无符号十六进制表示组成。该方法用得比较多,一般子类都有覆盖。

public String toString() {
        return getClass().getName() + "@" + Integer.toHexString(hashCode());
    }
三、getClass()方法
public final native Class getClass();

返回次Object的运行时类类型。不可重写,要调用的话,一般和getName()联合使用,如getClass().getName();

四、finalize()方法
protected void finalize() throws Throwable { }

该方法用于释放资源。因为无法确定该方法什么时候被调用,很少使用。Java允许在类中定义一个名为finalize()的方法。它的工作原理是:一旦垃圾回收器准备好释放对象占用的存储空间,将首先调用其finalize()方法。并且在下一次垃圾回收动作发生时,才会真正回收对象占用的内存。

关于垃圾回收,有三点需要记住:

  • 1、对象可能不被垃圾回收。只要程序没有濒临存储空间用完的那一刻,对象占用的空间就总也得不到释放。
  • 2、垃圾回收并不等于“析构”。
  • 3、垃圾回收只与内存有关。使用垃圾回收的唯一原因是为了回收程序不再使用的内存。

finalize()的用途:

  • 无论对象是如何创建的,垃圾回收器都会负责释放对象占据的所有内存。这就将对finalize()的需求限制到一种特殊情况,即通过某种创建对象方式以外的方式为对象分配了存储空间。
  • 不过这种情况一般发生在使用“本地方法”的情况下,本地方法是一种在Java中调用非Java代码的方式。
五、equals()方法
public boolean equals(Object obj) {
        return (this == obj);
    }

Object中的equals方法是直接判断this和obj本身的值是否相等,即用来判断调用equals的对象和形参obj所引用的对象是否是同一对象,

所谓同一对象就是指内存中同一块存储单元,如果this和obj指向的hi同一块内存对象,则返回true,如果this和obj指向的不是同一块内存,则返回false。

注意:即便是内容完全相等的两块不同的内存对象,也返回false。

如果是同一块内存,则object中的equals方法返回true,如果是不同的内存,则返回false

如果希望不同内存但相同内容的两个对象equals时返回true,则我们需要重写父类的equal方法

String类已经重写了object中的equals方法(这样就是比较内容是否相等了)

六、hashcode()方法
public native int hashCode();

返回该对象的哈希码值,该方法用于哈希查找,可以减少在查找中使用equals的次数,重写了equals方法一般都要重写hashCode方法。这个方法在一些具有哈希功能的Collection中用到。一般必须满足obj1.equals(obj2)==true。可以推出obj1.hash- Code()==obj2.hashCode(),但是hashCode相等不一定就满足equals。不过为了提高效率,应该尽量使上面两个条件接近等价。如果不重写hashcode(),在HashSet中添加两个equals的对象,会将两个对象都加入进去。

七、wait()方法
wait()

wait(long timeout)

wait(long timeout,int naos)

wait方法就是使当前线程等待,当前线程必须是该对象的拥有者(具有该对象的锁)。wait()方法一直等待,直到获得锁或者被中断。wait(long timeout)设定一个超时间隔,如果在规定时间内没有获得锁就返回。

调用该方法后当前线程进入睡眠状态,直到以下事件发生。

  • (1)其他线程调用了该对象的notify方法。
  • (2)其他线程调用了该对象的notifyAll方法。
  • (3)其他线程调用了interrupt中断该线程。
  • (4)时间间隔到了。

此时该线程就可以被调度了,如果是被中断的话就抛出一个InterruptedException异常。

八、notify()方法
public final native void notify();

该方法唤醒在该对象上等待的某个线程。

九、notifyAll()方法
public final native void notifyAll();

该方法唤醒在该对象上等待的所有线程。

博文参考

《JDK的源码分析》

关注
打赏
1657692014
查看更多评论
立即登录/注册

微信扫码登录

0.1931s