夯实Java基础系列9深入理解Class类和Object类

3次阅读

共计 18468 个字符,预计需要花费 47 分钟才能阅读完成。

目录

  • Java 中 Class 类及用法

    • Class 类原理
    • 如何获得一个 Class 类对象
    • 使用 Class 类的对象来生成目标类的实例
  • Object 类

    • 类构造器 public Object();
    • registerNatives()方法;
    • Clone()方法实现浅拷贝
    • getClass()方法
    • equals()方法
    • hashCode()方法;
    • toString()方法
    • wait() notify() notifAll()
    • finalize()方法
  • CLass 类和 Object 类的关系
  • 参考文章
  • 微信公众号

    • Java 技术江湖
    • 个人公众号:黄小斜

– Object 类

本系列文章将整理到我在 GitHub 上的《Java 面试指南》仓库,更多精彩内容请到我的仓库里查看

https://github.com/h2pl/Java-…

喜欢的话麻烦点下 Star 哈

文章首发于我的个人博客:

www.how2playlife.com

本文是微信公众号【Java 技术江湖】的《夯实 Java 基础系列博文》其中一篇,本文部分内容来源于网络,为了把本文主题讲得清晰透彻,也整合了很多我认为不错的技术博客内容,引用其中了一些比较好的博客文章,如有侵权,请联系作者。
该系列博文会告诉你如何从入门到进阶,一步步地学习 Java 基础知识,并上手进行实战,接着了解每个 Java 知识点背后的实现原理,更完整地了解整个 Java 技术体系,形成自己的知识框架。为了更好地总结和检验你的学习成果,本系列文章也会提供每个知识点对应的面试题以及参考答案。

如果对本系列文章有什么建议,或者是有什么疑问的话,也可以关注公众号【Java 技术江湖】联系作者,欢迎你参与本系列博文的创作和修订。

<!– more –>

Java 中 Class 类及用法

Java 程序在运行时,Java 运行时系统一直对所有的对象进行所谓的运行时类型标识,即所谓的 RTTI。

这项信息纪录了每个对象所属的类。虚拟机通常使用运行时类型信息选准正确方法去执行,用来保存这些类型信息的类是 Class 类。Class 类封装一个对象和接口运行时的状态,当装载类时,Class 类型的对象自动创建。

说白了就是:

Class 类也是类的一种,只是名字和 class 关键字高度相似。Java 是大小写敏感的语言。

Class 类的对象内容是你创建的类的类型信息,比如你创建一个 shapes 类,那么,Java 会生成一个内容是 shapes 的 Class 类的对象

Class 类的对象不能像普通类一样,以 new shapes() 的方式创建,它的对象只能由 JVM 创建,因为这个类没有 public 构造函数

    /*
     * Private constructor. Only the Java Virtual Machine creates Class objects.
     * This constructor is not used and prevents the default constructor being
     * generated.
     */
     // 私有构造方法,只能由 jvm 进行实例化
    private Class(ClassLoader loader) {
        // Initialize final field for classLoader.  The initialization value of non-null
        // prevents future JIT optimizations from assuming this final field is null.
        classLoader = loader;
    }

Class 类的作用是运行时提供或获得某个对象的类型信息,和 C ++ 中的 typeid()函数类似。这些信息也可用于反射。

Class 类原理

看一下 Class 类的部分源码

//Class 类中封装了类型的各种信息。在 jvm 中就是通过 Class 类的实例来获取每个 Java 类的所有信息的。public class Class 类 {
    Class aClass = null;

//    private EnclosingMethodInfo getEnclosingMethodInfo() {//        Object[] enclosingInfo = getEnclosingMethod0();
//        if (enclosingInfo == null)
//            return null;
//        else {//            return new EnclosingMethodInfo(enclosingInfo);
//        }
//    }

    /** 提供原子类操作
     * Atomic operations support.
     */
//    private static class Atomic {
//        // initialize Unsafe machinery here, since we need to call Class.class instance method
//        // and have to avoid calling it in the static initializer of the Class class...
//        private static final Unsafe unsafe = Unsafe.getUnsafe();
//        // offset of Class.reflectionData instance field
//        private static final long reflectionDataOffset;
//        // offset of Class.annotationType instance field
//        private static final long annotationTypeOffset;
//        // offset of Class.annotationData instance field
//        private static final long annotationDataOffset;
//
//        static {//            Field[] fields = Class.class.getDeclaredFields0(false); // bypass caches
//            reflectionDataOffset = objectFieldOffset(fields, "reflectionData");
//            annotationTypeOffset = objectFieldOffset(fields, "annotationType");
//            annotationDataOffset = objectFieldOffset(fields, "annotationData");
//        }

        // 提供反射信息
    // reflection data that might get invalidated when JVM TI RedefineClasses() is called
//    private static class ReflectionData<T> {//        volatile Field[] declaredFields;
//        volatile Field[] publicFields;
//        volatile Method[] declaredMethods;
//        volatile Method[] publicMethods;
//        volatile Constructor<T>[] declaredConstructors;
//        volatile Constructor<T>[] publicConstructors;
//        // Intermediate results for getFields and getMethods
//        volatile Field[] declaredPublicFields;
//        volatile Method[] declaredPublicMethods;
//        volatile Class<?>[] interfaces;
//
//        // Value of classRedefinedCount when we created this ReflectionData instance
//        final int redefinedCount;
//
//        ReflectionData(int redefinedCount) {
//            this.redefinedCount = redefinedCount;
//        }
//    }
        // 方法数组
//    static class MethodArray {//        // Don't add or remove methods except by add() or remove() calls.
//        private Method[] methods;
//        private int length;
//        private int defaults;
//
//        MethodArray() {//            this(20);
//        }
//
//        MethodArray(int initialSize) {//            if (initialSize < 2)
//                throw new IllegalArgumentException("Size should be 2 or more");
//
//            methods = new Method[initialSize];
//            length = 0;
//            defaults = 0;
//        }

    // 注解信息
    // annotation data that might get invalidated when JVM TI RedefineClasses() is called
//    private static class AnnotationData {
//        final Map<Class<? extends Annotation>, Annotation> annotations;
//        final Map<Class<? extends Annotation>, Annotation> declaredAnnotations;
//
//        // Value of classRedefinedCount when we created this AnnotationData instance
//        final int redefinedCount;
//
//        AnnotationData(Map<Class<? extends Annotation>, Annotation> annotations,
//                       Map<Class<? extends Annotation>, Annotation> declaredAnnotations,
//                       int redefinedCount) {
//            this.annotations = annotations;
//            this.declaredAnnotations = declaredAnnotations;
//            this.redefinedCount = redefinedCount;
//        }
//    }
}

我们都知道所有的 java 类都是继承了 object 这个类,在 object 这个类中有一个方法:getclass(). 这个方法是用来取得该类已经被实例化了的对象的该类的引用,这个引用指向的是 Class 类的对象。

我们自己无法生成一个 Class 对象(构造函数为 private),而 这个 Class 类的对象是在当各类被调入时,由 Java 虚拟机自动创建 Class 对象,或通过类装载器中的 defineClass 方法生成。

// 通过该方法可以动态地将字节码转为一个 Class 类对象
protected final Class<?> defineClass(String name, byte[] b, int off, int len)
    throws ClassFormatError
{return defineClass(name, b, off, len, null);
}

我们生成的对象都会有个字段记录该对象所属类在 CLass 类的对象的所在位置。如下图所示:

[外链图片转存失败(img-ZfMJTzO4-1569074134147)(http://dl.iteye.com/upload/pi…]

如何获得一个 Class 类对象

请注意,以下这些方法都是值、指某个类对应的 Class 对象已经在堆中生成以后,我们通过不同方式获取对这个 Class 对象的引用。而上面说的 DefineClass 才是真正将字节码加载到虚拟机的方法,会在堆中生成新的一个 Class 对象。

第一种办法,Class 类的 forName 函数

public class shapes{}
Class obj= Class.forName(“shapes”);
第二种办法,使用对象的 getClass()函数

public class shapes{}
shapes s1=new shapes();
Class obj=s1.getClass();
Class obj1=s1.getSuperclass();// 这个函数作用是获取 shapes 类的父类的类型

第三种办法,使用类字面常量

Class obj=String.class;
Class obj1=int.class;
注意,使用这种办法生成 Class 类对象时,不会使 JVM 自动加载该类(如 String 类)。== 而其他办法会使得 JVM 初始化该类。==

使用 Class 类的对象来生成目标类的实例

生成不精确的 object 实例

== 获取一个 Class 类的对象后,可以用 newInstance() 函数来生成目标类的一个实例。然而,该函数并不能直接生成目标类的实例,只能生成 object 类的实例 ==

Class obj=Class.forName(“shapes”);
Object ShapesInstance=obj.newInstance();
使用泛化 Class 引用生成带类型的目标实例

Class<shapes> obj=shapes.class;
shapes newShape=obj.newInstance();
因为有了类型限制,所以使用泛化 Class 语法的对象引用不能指向别的类。

Class obj1=int.class;
Class<Integer> obj2=int.class;
obj1=double.class;
//obj2=double.class; 这一行代码是非法的,obj2 不能改指向别的类

然而,有个灵活的用法,使得你可以用 Class 的对象指向基类的任何子类。Class<? extends Number> obj=int.class;
obj=Number.class;
obj=double.class;

因此,以下语法生成的 Class 对象可以指向任何类。Class<?> obj=int.class;
obj=double.class;
obj=shapes.class;
最后一个奇怪的用法是,当你使用这种泛型语法来构建你手头有的一个 Class 类的对象的基类对象时,必须采用以下的特殊语法

public class shapes{}
class round extends shapes{}
Class<round> rclass=round.class;
Class<? super round> sclass= rclass.getSuperClass();
//Class<shapes> sclass=rclass.getSuperClass();
我们明知道,round 的基类就是 shapes,但是却不能直接声明 Class < shapes >,必须使用特殊语法

Class < ? super round >

这个记住就可以啦。

Object 类

这部分主要参考 http://ihenu.iteye.com/blog/2…

Object 类是 Java 中其他所有类的祖先,没有 Object 类 Java 面向对象无从谈起。作为其他所有类的基类,Object 具有哪些属性和行为,是 Java 语言设计背后的思维体现。

Object 类位于 java.lang 包中,java.lang 包包含着 Java 最基础和核心的类,在编译时会自动导入。Object 类没有定义属性,一共有 13 个方法,13 个方法之中并不是所有方法都是子类可访问的,一共有 9 个方法是所有子类都继承了的。

先大概介绍一下这些方法

1.clone 方法
保护方法,实现对象的浅复制,只有实现了 Cloneable 接口才可以调用该方法,否则抛出 CloneNotSupportedException 异常。2.getClass 方法
final 方法,获得运行时类型。3.toString 方法
该方法用得比较多,一般子类都有覆盖。4.finalize 方法
该方法用于释放资源。因为无法确定该方法什么时候被调用,很少使用。5.equals 方法
该方法是非常重要的一个方法。一般 equals 和 == 是不一样的,但是在 Object 中两者是一样的。子类一般都要重写这个方法。6.hashCode 方法
该方法用于哈希查找,重写了 equals 方法一般都要重写 hashCode 方法。这个方法在一些具有哈希功能的 Collection 中用到。一般必须满足 obj1.equals(obj2)==true。可以推出 obj1.hash- Code()==obj2.hashCode(),但是 hashCode 相等不一定就满足 equals。不过为了提高效率,应该尽量使上面两个条件接近等价。7.wait 方法
wait 方法就是使当前线程等待该对象的锁,当前线程必须是该对象的拥有者,也就是具有该对象的锁。wait()方法一直等待,直到获得锁或者被中断。wait(long timeout)设定一个超时间隔,如果在规定时间内没有获得锁就返回。调用该方法后当前线程进入睡眠状态,直到以下事件发生。(1)其他线程调用了该对象的 notify 方法。(2)其他线程调用了该对象的 notifyAll 方法。(3)其他线程调用了 interrupt 中断该线程。(4)时间间隔到了。此时该线程就可以被调度了,如果是被中断的话就抛出一个 InterruptedException 异常。8.notify 方法
该方法唤醒在该对象上等待的某个线程。9.notifyAll 方法
该方法唤醒在该对象上等待的所有线程。

类构造器 public Object();

大部分情况下,Java 中通过形如 new A(args..)形式创建一个属于该类型的对象。其中 A 即是类名,A(args..)即此类定义中相对应的构造函数。通过此种形式创建的对象都是通过类中的构造函数完成。

为体现此特性,Java 中规定:在类定义过程中,对于未定义构造函数的类,默认会有一个无参数的构造函数,作为所有类的基类,Object 类自然要反映出此特性,在源码中,未给出 Object 类构造函数定义,但实际上,此构造函数是存在的。

当然,并不是所有的类都是通过此种方式去构建,也自然的,并不是所有的类构造函数都是 public。

registerNatives()方法;

private static native void registerNatives();

registerNatives 函数前面有 native 关键字修饰,Java 中,用 native 关键字修饰的函数表明该方法的实现并不是在 Java 中去完成,而是由 C /C++ 去完成,并被编译成了.dll,由 Java 去调用。

方法的具体实现体在 dll 文件中,对于不同平台,其具体实现应该有所不同。用 native 修饰,即表示操作系统,需要提供此方法,Java 本身需要使用。

具体到 registerNatives()方法本身,其主要作用是将 C /C++ 中的方法映射到 Java 中的 native 方法,实现方法命名的解耦。

既然如此,可能有人会问,registerNatives()修饰符为 private,且并没有执行,作用何以达到?其实,在 Java 源码中,此方法的声明后有紧接着一段静态代码块:

private static native void registerNatives();  
static {registerNatives();  
}  

Clone()方法实现浅拷贝

protected native Object clone() throwsCloneNotSupportedException;

看,clode()方法又是一个被声明为 native 的方法,因此,我们知道了 clone()方法并不是 Java 的原生方法,具体的实现是有 C /C++ 完成的。clone 英文翻译为 ” 克隆 ”,其目的是创建并返回此对象的一个副本。

形象点理解,这有一辆科鲁兹,你看着不错,想要个一模一样的。你调用此方法即可像变魔术一样变出一辆一模一样的科鲁兹出来。配置一样,长相一样。但从此刻起,原来的那辆科鲁兹如果进行了新的装饰,与你克隆出来的这辆科鲁兹没有任何关系了。

你克隆出来的对象变不变完全在于你对克隆出来的科鲁兹有没有进行过什么操作了。Java 术语表述为:clone 函数返回的是一个引用,指向的是新的 clone 出来的对象,此对象与原对象分别占用不同的堆空间。

明白了 clone 的含义后,接下来看看如果调用 clone()函数对象进行此克隆操作。

首先看一下下面的这个例子:

package com.corn.objectsummary;  
  
import com.corn.Person;  
  
public class ObjectTest {public static void main(String[] args) {Object o1 = new Object();  
        // The method clone() from the type Object is not visible  
        Object clone = o1.clone();}  
  
}  

例子很简单,在 main()方法中,new 一个 Oject 对象后,想直接调用此对象的 clone 方法克隆一个对象,但是出现错误提示:”The method clone() from the type Object is not visible”

why? 根据提示,第一反应是 ObjectTest 类中定义的 Oject 对象无法访问其 clone()方法。回到 Object 类中 clone()方法的定义,可以看到其被声明为 protected,估计问题就在这上面了,protected 修饰的属性或方法表示:在同一个包内或者不同包的子类可以访问。

显然,Object 类与 ObjectTest 类在不同的包中,但是 ObjectTest 继承自 Object,是 Object 类的子类,于是,现在却出现子类中通过 Object 引用不能访问 protected 方法,原因在于对 ” 不同包中的子类可以访问 ” 没有正确理解。

“ 不同包中的子类可以访问 ”,是指当两个类不在同一个包中的时候,继承自父类的子类内部且主调(调用者)为子类的引用时才能访问父类用 protected 修饰的成员(属性 / 方法)。在子类内部,主调为父类的引用时并不能访问此 protected 修饰的成员。!(super 关键字除外)

于是,上例改成如下形式,我们发现,可以正常编译:

    public class clone 方法 {public static void main(String[] args) { }
    public void test1() {User user = new User();
//        User copy = user.clone();}
    public void test2() {User user = new User();
//        User copy = (User)user.clone();}
}

是的,因为此时的主调已经是子类的引用了。

上述代码在运行过程中会抛出 ”java.lang.CloneNotSupportedException”, 表明 clone()方法并未正确执行完毕,问题的原因在与 Java 中的语法规定:

clone()的正确调用是需要实现 Cloneable 接口,如果没有实现 Cloneable 接口,并且子类直接调用 Object 类的 clone()方法,则会抛出 CloneNotSupportedException 异常。

Cloneable 接口仅是一个表示接口,接口本身不包含任何方法,用来指示 Object.clone()可以合法的被子类引用所调用。

于是,上述代码改成如下形式,即可正确指定 clone()方法以实现克隆。

public class User implements Cloneable{
public int id;
public String name;
public UserInfo userInfo;

public static void main(String[] args) {User user = new User();
    UserInfo userInfo = new UserInfo();
    user.userInfo = userInfo;
    System.out.println(user);
    System.out.println(user.userInfo);
    try {User copy = (User) user.clone();
        System.out.println(copy);
        System.out.println(copy.userInfo);
    } catch (CloneNotSupportedException e) {e.printStackTrace();
    }
}
// 拷贝的 User 实例与原来不一样,是两个对象。//    com.javase.Class 和 Object.Object 方法. 用到的类.User@4dc63996
//    com.javase.Class 和 Object.Object 方法. 用到的类.UserInfo@d716361
        // 而拷贝后对象的 userinfo 引用对象是同一个。// 所以这是浅拷贝
//    com.javase.Class 和 Object.Object 方法. 用到的类.User@6ff3c5b5
//    com.javase.Class 和 Object.Object 方法. 用到的类.UserInfo@d716361
}

总结:
clone 方法实现的是浅拷贝,只拷贝当前对象,并且在堆中分配新的空间,放这个复制的对象。但是对象如果里面有其他类的子对象,那么就不会拷贝到新的对象中。

== 深拷贝和浅拷贝的区别 ==

浅拷贝
浅拷贝是按位拷贝对象,它会创建一个新对象,这个对象有着原始对象属性值的一份精确拷贝。如果属性是基本类型,拷贝的就是基本类型的值;如果属性是内存地址(引用类型),拷贝的就是内存地址,因此如果其中一个对象改变了这个地址,就会影响到另一个对象。

深拷贝
深拷贝会拷贝所有的属性, 并拷贝属性指向的动态分配的内存。当对象和它所引用的对象一起拷贝时即发生深拷贝。深拷贝相比于浅拷贝速度较慢并且花销较大。
现在为了要在 clone 对象时进行深拷贝,那么就要 Clonable 接口,覆盖并实现 clone 方法,除了调用父类中的 clone 方法得到新的对象,还要将该类中的引用变量也 clone 出来。如果只是用 Object 中默认的 clone 方法,是浅拷贝的。

那么这两种方式有什么相同和不同呢?

new 操作符的本意是分配内存。程序执行到 new 操作符时,首先去看 new 操作符后面的类型,因为知道了类型,才能知道要分配多大的内存空间。

分配完内存之后,再调用构造函数,填充对象的各个域,这一步叫做对象的初始化,构造方法返回后,一个对象创建完毕,可以把他的引用(地址)发布到外部,在外部就可以使用这个引用操纵这个对象。

而 clone 在第一步是和 new 相似的,都是分配内存,调用 clone 方法时,分配的内存和源对象(即调用 clone 方法的对象)相同,然后再使用原对象中对应的各个域,填充新对象的域,

填充完成之后,clone 方法返回,一个新的相同的对象被创建,同样可以把这个新对象的引用发布到外部。

== 也就是说,一个对象在浅拷贝以后,只是把对象复制了一份放在堆空间的另一个地方,但是成员变量如果有引用指向其他对象,这个引用指向的对象和被拷贝的对象中引用指向的对象是一样的。当然,基本数据类型还是会重新拷贝一份的。==

getClass()方法

4.public final native Class<?> getClass();

getClass()也是一个 native 方法,返回的是此 Object 对象的类对象 / 运行时类对象 Class<?>。效果与 Object.class 相同。

首先解释下 ” 类对象 ” 的概念:在 Java 中,类是是对具有一组相同特征或行为的实例的抽象并进行描述,对象则是此类所描述的特征或行为的具体实例。

作为概念层次的类,其本身也具有某些共同的特性,如都具有类名称、由类加载器去加载,都具有包,具有父类,属性和方法等。

于是,Java 中有专门定义了一个类,Class,去描述其他类所具有的这些特性,因此,从此角度去看,类本身也都是属于 Class 类的对象。为与经常意义上的对象相区分,在此称之为 ” 类对象 ”。

public class getClass 方法 {public static void main(String[] args) {User user = new User();
        //getclass 方法是 native 方法,可以取到堆区唯一的 Class<User> 对象
        Class<?> aClass = user.getClass();
        Class bClass = User.class;
        try {Class cClass = Class.forName("com.javase.Class 和 Object.Object 方法. 用到的类.User");
        } catch (ClassNotFoundException e) {e.printStackTrace();
        }
        System.out.println(aClass);
        System.out.println(bClass);
//        class com.javase.Class 和 Object.Object 方法. 用到的类.User
//        class com.javase.Class 和 Object.Object 方法. 用到的类.User
        try {User a = (User) aClass.newInstance();} catch (InstantiationException e) {e.printStackTrace();
        } catch (IllegalAccessException e) {e.printStackTrace();
        }
    }
} 

此处主要大量涉及到 Java 中的反射知识

equals()方法

5.public boolean equals(Object obj);

与 equals 在 Java 中经常被使用,大家也都知道与 equals 的区别:

== 表示的是变量值完成相同(对于基础类型,地址中存储的是值,引用类型则存储指向实际对象的地址);

equals 表示的是对象的内容完全相同,此处的内容多指对象的特征 / 属性。

实际上,上面说法是不严谨的,更多的只是常见于 String 类中。首先看一下 Object 类中关于 equals()方法的定义:

public boolean equals(Object obj) {return (this == obj);  
}  

由此可见,Object 原生的 equals()方法内部调用的正是 ==,与 == 具有相同的含义。既然如此,为什么还要定义此 equals()方法?

equals()方法的正确理解应该是:判断两个对象是否相等。那么判断对象相等的标尺又是什么?

如上,在 object 类中,此标尺即为 ==。当然,这个标尺不是固定的,其他类中可以按照实际的需要对此标尺含义进行重定义。如 String 类中则是依据字符串内容是否相等来重定义了此标尺含义。如此可以增加类的功能型和实际编码的灵活性。当然了,如果自定义的类没有重写 equals()方法来重新定义此标尺,那么默认的将是其父类的 equals(),直到 object 基类。

如下场景的实际业务需求,对于 User bean,由实际的业务需求可知当属性 uid 相同时,表示的是同一个 User,即两个 User 对象相等。则可以重写 equals 以重定义 User 对象相等的标尺。

ObjectTest 中打印出 true,因为 User 类定义中重写了 equals()方法,这很好理解,很可能张三是一个人小名,张三丰才是其大名,判断这两个人是不是同一个人,这时只用判断 uid 是否相同即可。

如上重写 equals 方法表面上看上去是可以了,实则不然。因为它破坏了 Java 中的约定:重写 equals()方法必须重写 hasCode()方法。

hashCode()方法;

  1. public native int hashCode()

hashCode()方法返回一个整形数值,表示该对象的哈希码值。

hashCode()具有如下约定:

1). 在 Java 应用程序程序执行期间,对于同一对象多次调用 hashCode()方法时,其返回的哈希码是相同的,前提是将对象进行 equals 比较时所用的标尺信息未做修改。在 Java 应用程序的一次执行到另外一次执行,同一对象的 hashCode()返回的哈希码无须保持一致;

2). 如果两个对象相等(依据:调用 equals()方法),那么这两个对象调用 hashCode()返回的哈希码也必须相等;

3). 反之,两个对象调用 hasCode()返回的哈希码相等,这两个对象不一定相等。

即严格的数学逻辑表示为:两个对象相等 <=>  equals()相等  => hashCode()相等。因此,重写 equlas()方法必须重写 hashCode()方法,以保证此逻辑严格成立,同时可以推理出:hasCode()不相等 => equals()不相等 <=> 两个对象不相等。可能有人在此产生疑问:既然比较两个对象是否相等的唯一条件(也是冲要条件)是 equals,那么为什么还要弄出一个 hashCode(),并且进行如此约定,弄得这么麻烦?其实,这主要体现在 hashCode()方法的作用上,其主要用于增强哈希表的性能。以集合类中,以 Set 为例,当新加一个对象时,需要判断现有集合中是否已经存在与此对象相等的对象,如果没有 hashCode()方法,需要将 Set 进行一次遍历,并逐一用 equals()方法判断两个对象是否相等,此种算法时间复杂度为 o(n)。通过借助于 hasCode 方法,先计算出即将新加入对象的哈希码,然后根据哈希算法计算出此对象的位置,直接判断此位置上是否已有对象即可。(注:Set 的底层用的是 Map 的原理实现)

在此需要纠正一个理解上的误区:对象的 hashCode()返回的不是对象所在的物理内存地址。甚至也不一定是对象的逻辑地址,hashCode()相同的两个对象,不一定相等,换言之,不相等的两个对象,hashCode()返回的哈希码可能相同。

因此,在上述代码中,重写了 equals()方法后,需要重写 hashCode()方法。

public class equals 和 hashcode 方法 {
    @Override
    // 修改 equals 时必须同时修改 hashcode 方法,否则在作为 key 时会出问题
    public boolean equals(Object obj) {return (this == obj);
    }
    
    @Override
    // 相同的对象必须有相同 hashcode,不同对象可能有相同 hashcode
    public int hashCode() {return hashCode() >> 2;
    }
}

toString()方法

7.public String toString();

toString()方法返回该对象的字符串表示。先看一下 Object 中的具体方法体:public String toString() {return getClass().getName() + "@" + Integer.toHexString(hashCode());  
}  

toString()方法相信大家都经常用到,即使没有显式调用,但当我们使用 System.out.println(obj)时,其内部也是通过 toString()来实现的。

getClass()返回对象的类对象,getClassName()以 String 形式返回类对象的名称(含包名)。Integer.toHexString(hashCode())则是以对象的哈希码为实参,以 16 进制无符号整数形式返回此哈希码的字符串表示形式。

如上例中的 u1 的哈希码是 638,则对应的 16 进制为 27e,调用 toString()方法返回的结果为:com.corn.objectsummary.User@27e。

因此:toString()是由对象的类型和其哈希码唯一确定,同一类型但不相等的两个对象分别调用 toString()方法返回的结果可能相同。

wait() notify() notifAll()

8/9/10/11/12. wait(…) / notify() / notifyAll()

一说到 wait(…) / notify() | notifyAll()几个方法,首先想到的是线程。确实,这几个方法主要用于 java 多线程之间的协作。先具体看下这几个方法的主要含义:

wait():调用此方法所在的当前线程等待,直到在其他线程上调用此方法的主调(某一对象)的 notify()/notifyAll()方法。

wait(long timeout)/wait(long timeout, int nanos):调用此方法所在的当前线程等待,直到在其他线程上调用此方法的主调(某一对象)的 notisfy()/notisfyAll()方法,或超过指定的超时时间量。

notify()/notifyAll():唤醒在此对象监视器上等待的单个线程 / 所有线程。

wait(…) / notify() | notifyAll()一般情况下都是配套使用。下面来看一个简单的例子:

这是一个生产者消费者的模型,只不过这里只用 flag 来标识哪个线程需要工作

public class wait 和 notify {
    //volatile 保证线程可见性
    volatile static int flag = 1;
    //object 作为锁对象,用于线程使用 wait 和 notify 方法
    volatile static Object o = new Object();
    public static void main(String[] args) {new Thread(new Runnable() {
            @Override
            public void run() {
                //wait 和 notify 只能在同步代码块内使用
                synchronized (o) {while (true) {if (flag == 0) {
                            try {Thread.sleep(2000);
                                System.out.println("thread1 wait");
                                // 释放锁,线程挂起进入 object 的等待队列,后续代码运行
                                o.wait();} catch (InterruptedException e) {e.printStackTrace();
                            }
                        }
                        System.out.println("thread1 run");
                        System.out.println("notify t2");
                        flag = 0;
                        // 通知等待队列的一个线程获取锁
                        o.notify();}
                }
            }
        }).start();
        // 解释同上
        new Thread(new Runnable() {
            @Override
            public void run() {while (true) {synchronized (o) {if (flag == 1) {
                            try {Thread.sleep(2000);
                                System.out.println("thread2 wait");
                                o.wait();} catch (InterruptedException e) {e.printStackTrace();
                            }
                        }
                        System.out.println("thread2 run");
                        System.out.println("notify t1");
                        flag = 1;
                        o.notify();}
                }
            }
        }).start();}

    // 输出结果是
//    thread1 run
//    notify t2
//    thread1 wait
//    thread2 run
//    notify t1
//    thread2 wait
//    thread1 run
//    notify t2
// 不断循环
}

从上述例子的输出结果中可以得出如下结论:

1、wait(…)方法调用后当前线程将立即阻塞,且适当其所持有的同步代码块中的锁,直到被唤醒或超时或打断后且重新获取到锁后才能继续执行;

2、notify()/notifyAll()方法调用后,其所在线程不会立即释放所持有的锁,直到其所在同步代码块中的代码执行完毕,此时释放锁,因此,如果其同步代码块后还有代码,其执行则依赖于 JVM 的线程调度。

在 Java 源码中,可以看到 wait()具体定义如下:

public final void wait() throws InterruptedException {wait(0);  
}  

且 wait(long timeout, int nanos)方法定义内部实质上也是通过调用 wait(long timeout)完成。而 wait(long timeout)是一个 native 方法。因此,wait(…)方法本质上都是 native 方式实现。

notify()/notifyAll()方法也都是 native 方法。

Java 中线程具有较多的知识点,是一块比较大且重要的知识点。后期会有博文专门针对 Java 多线程作出详细总结。此处不再细述。

finalize()方法

  1. protected void finalize();

finalize 方法主要与 Java 垃圾回收机制有关。首先我们看一下 finalized 方法在 Object 中的具体定义:

protected void finalize() throws Throwable {}  

我们发现 Object 类中 finalize 方法被定义成一个空方法,为什么要如此定义呢?finalize 方法的调用时机是怎么样的呢?

首先,Object 中定义 finalize 方法表明 Java 中每一个对象都将具有 finalize 这种行为,其具体调用时机在:JVM 准备对此对形象所占用的内存空间进行垃圾回收前,将被调用。由此可以看出,此方法并不是由我们主动去调用的(虽然可以主动去调用,此时与其他自定义方法无异)。

CLass 类和 Object 类的关系

Object 类和 Class 类没有直接的关系。

Object 类是一切 java 类的父类,对于普通的 java 类,即便不声明,也是默认继承了 Object 类。典型的,可以使用 Object 类中的 toString()方法。

Class 类是用于 java 反射机制的,一切 java 类,都有一个对应的 Class 对象,他是一个 final 类。Class 类的实例表示,正在运行的 Java 应用程序中的类和接口。

转一个知乎很有趣的问题
https://www.zhihu.com/questio…

Java 的对象模型中:1 所有的类都是 Class 类的实例,Object 是类,那么 Object 也是 Class 类的一个实例。2 所有的类都最终继承自 Object 类,Class 是类,那么 Class 也继承自 Object。3 这就像是先有鸡还是先有蛋的问题,请问实际中 JVM 是怎么处理的?

这个问题中,第 1 个假设是错的:java.lang.Object 是一个 Java 类,但并不是 java.lang.Class 的一个实例。后者只是一个用于描述 Java 类与接口的、用于支持反射操作的类型。这点上 Java 跟其它一些更纯粹的面向对象语言(例如 Python 和 Ruby)不同。

而第 2 个假设是对的:java.lang.Class 是 java.lang.Object 的派生类,前者继承自后者。虽然第 1 个假设不对,但“鸡蛋问题”仍然存在:在一个已经启动完毕、可以使用的 Java 对象系统里,必须要有一个 java.lang.Class 实例对应 java.lang.Object 这个类;而 java.lang.Class 是 java.lang.Object 的派生类,按“一般思维”前者应该要在后者完成初始化之后才可以初始化…

事实是:这些相互依赖的核心类型完全可以在“混沌”中一口气都初始化好,然后对象系统的状态才叫做完成了“bootstrap”,后面就可以按照 Java 对象系统的一般规则去运行。JVM、JavaScript、Python、Ruby 等的运行时都有这样的 bootstrap 过程。

在“混沌”(boostrap 过程)里,JVM 可以为对象系统中最重要的一些核心类型先分配好内存空间,让它们进入 [已分配空间] 但[尚未完全初始化]状态。此时这些对象虽然已经分配了空间,但因为状态还不完整所以尚不可使用。

然后,通过这些分配好的空间把这些核心类型之间的引用关系串好。到此为止所有动作都由 JVM 完成,尚未执行任何 Java 字节码。然后这些核心类型就进入了 [完全初始化] 状态,对象系统就可以开始自我运行下去,也就是可以开始执行 Java 字节码来进一步完成 Java 系统的初始化了。

参考文章

https://www.cnblogs.com/congs…
https://www.jb51.net/article/…
https://blog.csdn.net/dufufd/…
https://blog.csdn.net/farsigh…
https://blog.csdn.net/xiaomin…

微信公众号

Java 技术江湖

如果大家想要实时关注我更新的文章以及分享的干货的话,可以关注我的公众号【Java 技术江湖】一位阿里 Java 工程师的技术小站,作者黄小斜,专注 Java 相关技术:SSM、SpringBoot、MySQL、分布式、中间件、集群、Linux、网络、多线程,偶尔讲点 Docker、ELK,同时也分享技术干货和学习经验,致力于 Java 全栈开发!

Java 工程师必备学习资源: 一些 Java 工程师常用学习资源,关注公众号后,后台回复关键字 “Java” 即可免费无套路获取。

个人公众号:黄小斜

作者是 985 硕士,蚂蚁金服 JAVA 工程师,专注于 JAVA 后端技术栈:SpringBoot、MySQL、分布式、中间件、微服务,同时也懂点投资理财,偶尔讲点算法和计算机理论基础,坚持学习和写作,相信终身学习的力量!

程序员 3T 技术学习资源: 一些程序员学习技术的资源大礼包,关注公众号后,后台回复关键字 “资料” 即可免费无套路获取。

正文完
 0