第三篇文章主要講解相關(guān)“ equals 與 == ”面試題分析,分為以下部分:
- String類(lèi)的“ equals 與 == ”
- Integer類(lèi)的“ equals 與 == ”
String類(lèi)的“ equals 與 == ”
在Java中對(duì)于字符串的創(chuàng)建,有兩種形式:字符串常量和使用 new 來(lái)構(gòu)造一個(gè) String 對(duì)象。
字面量形式
在JVM中,為了減少對(duì)字符串變量的重復(fù)創(chuàng)建,其擁有一段特殊的內(nèi)存空間,這段內(nèi)存被稱(chēng)為字符串常量池。
比如 String str1 = "Hello"
創(chuàng)建了一個(gè)對(duì)象。此時(shí)默認(rèn)字符串常量池中沒(méi)有內(nèi)容為“Hello”的對(duì)象存在。當(dāng)JVM在字符串常量池中沒(méi)有找到內(nèi)容為 "Hello" 的對(duì)象時(shí),就會(huì)創(chuàng)建一個(gè)內(nèi)容為 "Hello" 的對(duì)象,并將它的引用返回給變量a。
當(dāng)我們?cè)俅?String str2 = "Hello"
時(shí),會(huì)發(fā)生什么情況呢?
首先JVM會(huì)在字符串常量池中查詢(xún)是否有內(nèi)容為 "Hello" 的對(duì)象。因?yàn)榇藭r(shí)字符串常量池中已經(jīng)有內(nèi)容為 "Hello" 的對(duì)象了,故JVM不會(huì)創(chuàng)建新的地址空間,而是將原有的 "Hello" 對(duì)象的引用返回,所以此時(shí)變量 num1 和變量 num2 擁有的是同一個(gè)對(duì)象引用,即指向的是同一塊內(nèi)存地址。
使用 new 構(gòu)造 String 對(duì)象
當(dāng)我們 new 一個(gè)字符串對(duì)象時(shí),不管字符串常量池中是否有內(nèi)容相同的對(duì)象,JVM 都會(huì)創(chuàng)造一個(gè)新的對(duì)象,其內(nèi)存地址均不同。
String str = new String("Hello")
創(chuàng)建過(guò)程分析:當(dāng)執(zhí)行String str = new String("Hello")
;時(shí),JVM 首先在字符串常量池中查看是否存在字符串對(duì)象 "Hello",
- 如果不存在該對(duì)象,則先在字符串常量池中創(chuàng)建一個(gè)新的字符串對(duì)象"Hello",然后執(zhí)行
new String("Hello")
構(gòu)造方法,在堆內(nèi)存中創(chuàng)建一個(gè)新的字符串對(duì)象"Hello",并將引用 str 指向堆內(nèi)存中創(chuàng)建的新對(duì)象; - 如果已存在該對(duì)象,則不用創(chuàng)建新的字符串對(duì)象"Hello",而直接使用字符串常量池中已存在的對(duì)象"Hello", 然后執(zhí)行
new String("Hello")
構(gòu)造方法,在堆內(nèi)存里創(chuàng)建一個(gè)新的字符串對(duì)象 "Hello",并將引用 str 指向堆內(nèi)存中創(chuàng)建的新對(duì)象。
測(cè)試代碼:
public class EqualsTest {
/**
* @param args
*/
public static void main(String[] args) {
String str1 = new String("Hello");
String str2 = new String("Hello");
String str3 = "Hello";
String str4 = "Hel" + "lo";
System.out.println("====== String equals ======");
System.out.println("str1 == str2 : " + (str1 == str2));
System.out.println("str1 == str3 : " + (str1 == str3));
System.out.println("str1.equals(str2) : " + str1.equals(str2));
System.out.println("str1.equals(str3) : " + str1.equals(str3));
System.out.println("str1 == str4 : " + (str1 == str4));
System.out.println("str1.equals(str4) : " + str1.equals(str4));
}
}
測(cè)試結(jié)果:
====== String equals ======
str1 == str2 : false
str1 == str3 : false
str1.equals(str2) : true
str1.equals(str3) : true
str1 == str4 : false
str1.equals(str4) : true
Integer 類(lèi)的“ equals 與 == ”
自動(dòng)裝箱與自動(dòng)拆箱
自動(dòng)裝箱:將Java的基本類(lèi)型轉(zhuǎn)換成包裝器類(lèi)型
自動(dòng)拆箱:將Java的包裝器類(lèi)型轉(zhuǎn)換成基本類(lèi)型
Java數(shù)據(jù)類(lèi)型分為基本數(shù)據(jù)類(lèi)型和包裝器類(lèi)型。
基本類(lèi)型: short int long float double byte boolean char
包裝器類(lèi)型: Short Integer Long Float Double Byte Boolean Character
Integer num = 1 // 自動(dòng)裝箱
Java調(diào)用 Integer 的 valueOf(int) 方法來(lái)完成自動(dòng)裝箱的
int num1 = num // 自動(dòng)拆箱
Java調(diào)用 Integer 的 intValue 方法來(lái)完成自動(dòng)拆箱的
注意:對(duì)于 Integer 類(lèi)型的對(duì)象來(lái)說(shuō),如果我們要?jiǎng)?chuàng)建的 Integer 對(duì)象的數(shù)值在 [-128,127]的區(qū)間之內(nèi),緩存中早已存儲(chǔ)了該范圍的值,那么JVM就會(huì)在緩存中查找,返回已經(jīng)存在的對(duì)象的引用;超過(guò)此范圍,JVM需要重新創(chuàng)建該對(duì)象。
測(cè)試代碼:
public class EqualsTest {
/**
* @param args
*/
public static void main(String[] args) {
int num1 = 1;
Integer num2 = 1;
Integer num3 = 1;
Integer num4 = 128;
Integer num5 = 128;
System.out.println("====== Integer equals ======");
System.out.println("num1 == num2 : " + (num1 == num2));
System.out.println("num2.equals(num1) : " + num2.equals(num1));
System.out.println("num2 == num3 : " + (num2 == num3));
System.out.println("num2.equals(num3) : " + num2.equals(num3));
System.out.println("num4 == num5 : " + (num4 == num5));
System.out.println("num4.equals(num5) : " + num4.equals(num5));
}
}
測(cè)試結(jié)果:
====== Integer equals ======
num1 == num2 : true
num2.equals(num1) : true
num2 == num3 : true
num2.equals(num3) : true
num4 == num5 : false
num4.equals(num5) : true