如图可知,Object类有12个成员方法,按照用途可以分为以下几种
1,构造函数
2,hashCode和equals函数用来判断对象是否相同,
3,wait(),wait(long),wait(long,int),notify(),notifyAll()
4,toString()和getClass,
5,clone()
6,finalize()用于在垃圾回收
函数说明
clone()
clone()函数的用途是用来另存一个当前存在的对象。(克隆)
hashCode()和equals()
- equals()用于确认两个对象是否相同。
- hashCode()用于获取对象的哈希值。
- 哈希值相同的对象不一定equals(),equals()返回true的两个对象一定相同。
Java中Object类的equals()和hashCode()方法深入解析
1.equals()
在初学Java的时候,很多人会说在比较对象的时候,==是比较地址,equals()是比较对象的内容,谁说的?
看看equals()方法在Object类中的定义:
public boolean equals(Object obj){
return (this == obj);
}
但是为什么会有equals是比较内容的这种说法呢?
因为在String、Double等封装类中,已经重写(overriding)了Object类的equals()方法,于是有了另一种计算公式,是进行内容的比较。
比如在String类中:
public int hashCode() {
int h = hash;
if (h == 0) {
char val[] = value;
int len = count;
for (int i = 0; i < len; i++) {
h = 31*h + val[off++];
}
hash = h;
}
return h;
}
2.hashCode()
在Object类中的定义为:
public native int hashCode();
是一个本地方法,返回的对象的地址值。
但是,同样的思路,在String等封装类中对此方法进行了重写。方法调用得到一个计算公式得到的int值。
3.两者的关系
①两个obj,如果equals()相等,hashCode()一定相等
②两个obj,如果hashCode()相等,equals()不一定相等
原因:从散列的角度考虑,不同的对象计算哈希码的时候,可能引起冲突,大家一定还记得数据结构中冲突的解决方案吧。
但是为什么要这样设计这两个函数,个人的理解是为了比较两个对象时更高效。
可以考虑在Java集合中,判断两个对象是否相等的规则是:
第一步,如果hashCode()相等,则查看第二步,否则不相等;
第二步,查看equals()是否相等,如果相等,则两obj相等,否则还是不相等。
为什么这样做?个人的理解是让适当的函数完成适当的功能,毕竟hashCode()比equals()在某种程度上来得快。
toString()和getClass()
toString()返回一个String对象,将信息变为字符串返回,默认输出对象地址,若输出对象的地址,我们并不能看懂它的意思,那么就要通过重写Object类的toString方法来输出对象属性信息。
getClass()返回此Object的运行时类,返回的类对象是被表示类的static synchronized方法锁定的对象。
返回一个Class对象如果打印出来会发现结果是如下格式
class package.name.xxx
因为返回的是一个class对象,后面可以跟class类的方法。用的是谁的构造函数,那么getClass返回的就是谁的类型。
getClass()经常用于java反射机制
wait(),wait(long),wait(long,int),notify(),notifyAll()
² 这几个函数体现的是Java的多线程机制
² 在使用的时候要求在synchronize语句中使用
² wait()用于让当前线程失去操作权限,当前线程进入等待序列
² notify()用于随机通知一个持有对象的锁的线程获取操作权限
² notifyAll()用于通知所有持有对象的锁的线程获取操作权限
² wait(long) 和wait(long,int)用于设定下一次获取锁的距离当前释放锁的时间间隔
notify和notifyAll的区别 - 何时使用
参考地址:https://blog.csdn.net/u014658905/article/details/81035870
Java中notify和notifyAll的区别
Java提供了两个方法notify和notifyAll来唤醒在某些条件下等待的线程,你可以使用它们中的任何一个,但是Java中的notify和notifyAll之间存在细微差别,这使得它成为Java中流行的多线程面试问题之一。
当你调用notify时,只有一个等待线程会被唤醒而且它不能保证哪个线程会被唤醒,这取决于线程调度器。虽然如果你调用notifyAll方法,那么等待该锁的所有线程都会被唤醒,但是在执行剩余的代码之前,所有被唤醒的线程都将争夺锁定,这就是为什么在循环上调用wait,因为如果多个线程被唤醒,那么线程是将获得锁定将首先执行,它可能会重置等待条件,这将迫使后续线程等待。
因此,notify和notifyAll之间的关键区别在于notify()只会唤醒一个线程,而notifyAll方法将唤醒所有线程。
何时在Java中使用notify和notifyAll
如果所有线程都在等待相同的条件,并且一次只有一个线程可以从条件变为true,则可以使用notify or notifyAll。
在这种情况下,notify是优于notifyAll因为唤醒所有这些因为我们知道只有一个线程会受益而所有其他线程将再次等待,所以调用notifyAll方法只是浪费CPU。
虽然这看起来很合理,但仍有一个警告,即无意中的接收者吞下了关键通知。通过使用notifyAll,我们确保所有收件人都会收到通知
总结:
1)如果我使用notify(),将通知哪个线程?
无法保证,ThreadScheduler将从等待该监视器上的线程的池中选择一个随机线程。保证只有一个线程会被通知(随机性)
2) 我怎么知道有多少线程在等待,所以我可以使用notifyAll()?
它取决于程序逻辑,在编码时需要考虑一段代码是否可以由多个线程运行。理解线程间通信的一个很好的例子是在Java中实现生产者 - 消费者模式。
3) 如何调用notify()?
Wait()和notify()方法只能从synchronized方法或块中调用,需要在其他线程正在等待的对象上调用notify方法。
4) 什么是这些线程等待被通知等?
线程等待某些条件,例如在生产者 - 消费者问题中,如果共享队列已满,则生产者线程等待,如果共享队列为空,则生成者线程等待。由于多个线程正在使用共享资源,因此它们使用wait和notify方法相互通信。
finalize()
当垃圾收集确定不再有对该对象的引用时,垃圾收集器在对象上调用该对象。
这个函数在进行垃圾回收的时候会用到,匿名对象回收之前会调用到,具体的例子如图所示