java中的==、equals()、hashCode()源码分析(转载)
在java编程或者面试中经常会遇到 == 、equals()的比较。自己看了看源码,结合实际的编程总结一下。
1. ==
java中的==是比较两个对象在JVM中的地址。比较好理解。看下面的代码:
1 public class ComAddr{ 2 public static void main(String[] args) throws Exception { 3 String s1 = "nihao"; 4 String s2 = "nihao"; 5 String s3 = new String("nihao"); 6 System.out.println(s1 == s2); // true 7 System.out.println(s1 == s3); // false 8 } 9 }
上述代码中:
(1)s1 == s2为true,是因为s1和s2都是字符串字面值"nihao"的引用,指向同一块地址,所以相等。
(2)s1 == s3为false,是因为通过new产生的对象在堆中,s3是堆中变量的引用,而是s1是指向字符串字面值"nihao"的引用,地址不同所以不相等。
2.equals()
equals是根类Obeject中的方法。源代码如下:
public boolean equals(Object obj) {return (this == obj); }
可见默认的equals方法,直接调用==,比较对象地址。
不同的子类,可以重写此方法,进行两个对象的equals的判断。
String类源码中重写的equals方法如下,
1 public boolean equals(Object anObject) {2 if (this == anObject) {3 return true;4 }5 if (anObject instanceof String) {6 String anotherString = (String) anObject;7 int n = value.length;8 if (n == anotherString.value.length) {9 char v1[] = value; 10 char v2[] = anotherString.value; 11 int i = 0; 12 while (n-- != 0) { 13 if (v1[i] != v2[i]) 14 return false; 15 i++; 16 } 17 return true; 18 } 19 } 20 return false; 21 }
从上面的代码中可以看到,
(1)String类中的equals首先比较地址,如果是同一个对象的引用,可知对象相等,返回true。
(2)若果不是同一个对象,equals方法挨个比较两个字符串对象内的字符,只有完全相等才返回true,否则返回false。
3.hashcode()
hashCode是根类Obeject中的方法。
默认情况下,Object中的hashCode() 返回对象的32位jvm内存地址。也就是说如果对象不重写该方法,则返回相应对象的32为JVM内存地址。
String类源码中重写的hashCode方法如下,
1 public int hashCode() {2 int h = hash; //Default to 0 ### String类中的私有变量,3 if (h == 0 && value.length > 0) { //private final char value[]; ### Sting类中保存的字符串内容的的数组4 char val[] = value;5 6 for (int i = 0; i < value.length; i++) {7 h = 31 * h + val[i];8 }9 hash = h; 10 } 11 return h; 12 }
String源码中使用private final char value[];保存字符串内容,因此String是不可变的。
看下面的例子,没有重写hashCode方法的类,直接返回32位对象在JVM中的地址;Long类重写了hashCode方法,返回计算出的hashCode数值:
1 public class ComHashcode{2 public static void main(String[] args) throws Exception {3 ComHashcode a = new ComHashcode();4 ComHashcode b = new ComHashcode();5 System.out.println(a.hashCode()); //8709196966 System.out.println(b.hashCode()); //2987927207 8 Long num1 = new Long(8);9 Long num2 = new Long(8); 10 System.out.println(num1.hashCode()); //8 11 System.out.println(num2.hashCode()); //8 12 } 13 }
总结:
(1)绑定。当equals方法被重写时,通常有必要重写 hashCode 方法,以维护 hashCode 方法的常规协定,该协定声明相等对象必须具有相等的哈希码。
(2)绑定原因。Hashtable实现一个哈希表,为了成功地在哈希表中存储和检索对象,用作键的对象必须实现 hashCode
方法和 equals
方法。同(1),必须保证equals相等的对象,hashCode
也相等。因为哈希表通过hashCode检索对象。
(3)默认。
==默认比较对象在JVM中的地址。
hashCode 默认返回对象在JVM中的存储地址。
equal比较对象,默认也是比较对象在JVM中的地址,同==
参考:
http://docs.oracle.com/javase/7/docs/api/
java中的==、equals()、hashCode()源码分析
在java编程或者面试中经常会遇到 == 、equals()的比较。自己看了看源码,结合实际的编程总结一下。
1. ==
java中的==是比较两个对象在JVM中的地址。比较好理解。看下面的代码:
1 public class ComAddr{ 2 public static void main(String[] args) throws Exception { 3 String s1 = "nihao"; 4 String s2 = "nihao"; 5 String s3 = new String("nihao"); 6 System.out.println(s1 == s2); // true 7 System.out.println(s1 == s3); // false 8 } 9 }
上述代码中:
(1)s1 == s2为true,是因为s1和s2都是字符串字面值"nihao"的引用,指向同一块地址,所以相等。
(2)s1 == s3为false,是因为通过new产生的对象在堆中,s3是堆中变量的引用,而是s1是指向字符串字面值"nihao"的引用,地址不同所以不相等。
2.equals()
equals是根类Obeject中的方法。源代码如下:
public boolean equals(Object obj) {return (this == obj); }
可见默认的equals方法,直接调用==,比较对象地址。
不同的子类,可以重写此方法,进行两个对象的equals的判断。
String类源码中重写的equals方法如下,
1 public boolean equals(Object anObject) {2 if (this == anObject) {3 return true;4 }5 if (anObject instanceof String) {6 String anotherString = (String) anObject;7 int n = value.length;8 if (n == anotherString.value.length) {9 char v1[] = value; 10 char v2[] = anotherString.value; 11 int i = 0; 12 while (n-- != 0) { 13 if (v1[i] != v2[i]) 14 return false; 15 i++; 16 } 17 return true; 18 } 19 } 20 return false; 21 }
从上面的代码中可以看到,
(1)String类中的equals首先比较地址,如果是同一个对象的引用,可知对象相等,返回true。
(2)若果不是同一个对象,equals方法挨个比较两个字符串对象内的字符,只有完全相等才返回true,否则返回false。
3.hashcode()
hashCode是根类Obeject中的方法。
默认情况下,Object中的hashCode() 返回对象的32位jvm内存地址。也就是说如果对象不重写该方法,则返回相应对象的32为JVM内存地址。
String类源码中重写的hashCode方法如下,
1 public int hashCode() {2 int h = hash; //Default to 0 ### String类中的私有变量,3 if (h == 0 && value.length > 0) { //private final char value[]; ### Sting类中保存的字符串内容的的数组4 char val[] = value;5 6 for (int i = 0; i < value.length; i++) {7 h = 31 * h + val[i];8 }9 hash = h; 10 } 11 return h; 12 }
String源码中使用private final char value[];保存字符串内容,因此String是不可变的。
看下面的例子,没有重写hashCode方法的类,直接返回32位对象在JVM中的地址;Long类重写了hashCode方法,返回计算出的hashCode数值:
1 public class ComHashcode{2 public static void main(String[] args) throws Exception {3 ComHashcode a = new ComHashcode();4 ComHashcode b = new ComHashcode();5 System.out.println(a.hashCode()); //8709196966 System.out.println(b.hashCode()); //2987927207 8 Long num1 = new Long(8);9 Long num2 = new Long(8); 10 System.out.println(num1.hashCode()); //8 11 System.out.println(num2.hashCode()); //8 12 } 13 }
总结:
(1)绑定。当equals方法被重写时,通常有必要重写 hashCode 方法,以维护 hashCode 方法的常规协定,该协定声明相等对象必须具有相等的哈希码。
(2)绑定原因。Hashtable实现一个哈希表,为了成功地在哈希表中存储和检索对象,用作键的对象必须实现 hashCode
方法和 equals
方法。同(1),必须保证equals相等的对象,hashCode
也相等。因为哈希表通过hashCode检索对象。
(3)默认。
==默认比较对象在JVM中的地址。
hashCode 默认返回对象在JVM中的存储地址。
equal比较对象,默认也是比较对象在JVM中的地址,同==
参考:
http://docs.oracle.com/javase/7/docs/api/\
http://www.cnblogs.com/xudong-bupt/p/3960177.html
转载于:https://www.cnblogs.com/GISQZC/p/5770298.html
java中的==、equals()、hashCode()源码分析(转载)相关推荐
- java中Mark接口_JVM源码分析之Java对象头实现
原标题:JVM源码分析之Java对象头实现 原创申明:本文由公众号[猿灯塔]原创,转载请说明出处标注 "365篇原创计划"第十一篇. 今天呢!灯塔君跟大家讲: JVM源码分析之Ja ...
- java中jcl,spring-jcl 日志源码分析
1.spring-jcl介绍 JCL全称:Jakarta Commons Logging spring-jcl 采用了设计模式中的"适配器模式",它对外提供统一的接口,然后在适配类 ...
- Hhadoop-2.7.0中HDFS写文件源码分析(二):客户端实现(1)
一.综述 HDFS写文件是整个Hadoop中最为复杂的流程之一,它涉及到HDFS中NameNode.DataNode.DFSClient等众多角色的分工与合作. 首先上一段代码,客户端是如何写文件的: ...
- 【kafka】Kafka中的动态配置源码分析
1.概述 2.源码分析 Broker启动加载动态配置 KafkaServer.startup 启动加载动态配置总流程 2.1 动态配置初始化 config.dynamicConfig.initiali ...
- suricata中DPDK收发包源码分析2
<suricata中DPDK收发包源码分析1>中分析了整体的DPDK收发包框架代码,今天我们继续来深入了解一下一些细节方面的问题. 目录 Q1:收发包线程模式在代码中是怎样确定的? Q2: ...
- Java类集框架 —— HashMap源码分析
HashMap是基于Map的键值对映射表,底层是通过数组.链表.红黑树(JDK1.8加入)来实现的. HashMap结构 HashMap中存储元素,是将key和value封装成了一个Node,先以一个 ...
- Java集合篇:LinkedList源码分析
(注:本文内容基于JDK1.6) 一.概述: LinkedList与ArrayList一样实现List接口,只是ArrayList是List接口的大小可变数组的实现,LinkedList是List接口 ...
- Java并发编程之ThreadLocal源码分析
1 一句话概括ThreadLocal 什么是ThreadLocal?顾名思义:线程本地变量,它为每个使用该对象的线程创建了一个独立的变量副本. 2 ThreadLocal使用场景 用一句话总结 ...
- 死磕java并发cas_死磕 java并发包之AtomicInteger源码分析
问题 (1)什么是原子操作? (2)原子操作和数据库的ACID有啥关系? (3)AtomicInteger是怎么实现原子操作的? (4)AtomicInteger是有什么缺点? 简介 AtomicIn ...
最新文章
- [Google API](8)构建使用Google API的Java应用程序
- 计算机中的进制和编码
- 关于新的描述语言GEZEL的介绍
- Linux rpm命令详解
- USACO 2.1 海明码(DFS)
- Android-Universal-Image-Loader学习笔记(二)--LruDiscCache
- 第88课:Spark Streaming从Flume Pull数据案例实战及内幕源码解密
- You-Get—— 基于 Python3 的媒体下载工具
- 更改wordpress上传文件大小限制
- 计算机一级主要学什么,计算机一级考试内容是什么 考试难不难
- 8个正弦波逆变器带你感受生活中无处不在的科技魅力
- jquery api的整体解读
- HTML实现倒计时插件
- Dinic算法的原理与构造
- Android 根据网络图片URL转Bitmap对象
- android 窗口切换花屏,分享Android4平台二级页面滚动花屏问题的解决方案v1.0.0
- 微信前台WEUI前台实例+EXCEL导出,SSM框架完整流程
- 《游戏机制——高级游戏设计技术》一1.1 规则定义游戏
- html img路径 404,webpack css背景图片无法现实,图片路径404
- 青海电大随学随考计算机,[青海电大]17秋随学随考心理学作业4题目
热门文章
- 需求分析用“试纸测试”的好处
- 使用C#编写一个求解数独的小软件
- Azure Key Vault(3):Key和Secret的区别
- CentOS7安装Nginx及配置域名映射反向代理
- NextCloud 最新官方源代码安装包及客户端下载
- Ubuntu下bundle install报错
- 基于Android的移动聊天APP设计与实现
- 计算机毕业设计ssm农商行贷款管理系统f1gnk系统+程序+源码+lw+远程部署
- 论文笔记 ACL 2019|Rapid Customization for Event Extraction
- 测试工程师如何拿高薪?