淺談Java編程中string的理解與運(yùn)用
一,“==”與equals()
運(yùn)行以下代碼,如何解釋其輸出結(jié)果?
public class StringPool { public static void main(String args[]) { String s0="Hello"; String s1="Hello"; String s2="He"+"llo"; System.out.println(s0==s1);//true System.out.println(s0==s2);//true System.out.println(new String("Hello")==new String("Hello"));//false } }
首先s0==s1
在Java執(zhí)行時(shí)會(huì)維護(hù)一個(gè)String堆,對(duì)于一些可以共享的字符串對(duì)象,會(huì)先在堆中查找是否存在相同的String內(nèi)容(字符相同),如果有就直接返回,不創(chuàng)建新對(duì)象。
s0中的值是引用的s1的值,自己并沒(méi)有創(chuàng)建對(duì)象,所以比較后的結(jié)果是true。
同理,s2中的值也是引用S1的值,所以比較的結(jié)果也是true
new String("Hello")==new String("Hello")
同時(shí)在堆中new了兩個(gè)對(duì)象,這兩個(gè)對(duì)象的內(nèi)容都是Hello,
但就好比a籃子和b籃子都裝了一個(gè)蘋果,a籃子裝了蘋果后和b籃子裝了蘋果后能判相等嗎?
當(dāng)然不行,假設(shè)蘋果都是一樣的,那蘋果當(dāng)然能和蘋果相等,但是籃子卻是不一樣的
在Java中,內(nèi)容相同的字串常量(“Hello”)只保存一份以節(jié)約內(nèi)存,所以s0,s1,s2實(shí)際上引用的是同一個(gè)對(duì)象。
編譯器在編譯s2一句時(shí),會(huì)去掉“+”號(hào),直接把兩個(gè)字串連接起來(lái)得一個(gè)字串(“Hello”)。這種優(yōu)化工作由Java編譯器自動(dòng)完成。
當(dāng)直接使用new關(guān)鍵字創(chuàng)建字符串對(duì)象時(shí),雖然值一致(都是“Hello”),但仍然是兩個(gè)獨(dú)立的對(duì)象。
Java中“==”的使用
1基本數(shù)據(jù)類型:比較的是內(nèi)容;
2引用數(shù)據(jù)類型:比較的是對(duì)象地址;
再看以下代碼
public static void main(String args[]) { String s1="a"; String s2=s1; System.out.println(s1==s2);//true s1+="b"; System.out.println(s1==s2);//false System.out.println(s1=="ab");//false System.out.println(s1.equals("ab"));//true }
分析:
給字串變量賦值意味著:兩個(gè)變量(s1,s2)現(xiàn)在引用同一個(gè)字符串對(duì)象“a”!
String對(duì)象的內(nèi)容是只讀的,使用“+”修改s1變量的值,實(shí)際上是得到了一個(gè)新的字符串對(duì)象,其內(nèi)容為“ab”,它與原先s1
所引用的對(duì)象”a”無(wú)關(guān),所以,s1==s2返回false;
代碼中的“ab”字符串是一個(gè)常量,它所引用的字符串與s1所引用的“ab”對(duì)象無(wú)關(guān)。
String.equals()方法可以比較兩個(gè)字符串的內(nèi)容。
二,String,equals()
方法
java中的String.equals()方法的實(shí)現(xiàn)代碼:
equals()法是根類Object中的方法。源代碼如下:
public boolean equals(Object obj) { return (this == obj); } //可見(jiàn)默認(rèn)的equals方法,直接調(diào)用==,比較對(duì)象地址。 // //不同的子類,可以重寫此方法,進(jìn)行兩個(gè)對(duì)象的equals的判斷。 //String類源碼中重寫的equals()方法的實(shí)現(xiàn)代碼如下: public boolean equals(Object anObject) { if(this==anObject) return true; if(anObject instanceof String) { String anotherString=(String)anObject; int n=value.length; if(n==anotherString.value.length) //若兩個(gè)字符串長(zhǎng)度一樣,則一個(gè)個(gè)進(jìn)行字符比較 { char v1[]=value;//字符串轉(zhuǎn)化成的對(duì)應(yīng)數(shù)組 char v2[]=anotherString.value; //字符串轉(zhuǎn)化成的對(duì)應(yīng)數(shù)組 int i=0; while(n--!=0) { if(v1[i]!=v2[i]) return false; //若比較過(guò)程中出現(xiàn)不等,則倆字符串不等,返回false i++; } return true; //直至比較完兩個(gè)字符串長(zhǎng)度,跳出while循環(huán) // 此時(shí)說(shuō)明倆字符串相等,返回true } } return false; //兩個(gè)字符串長(zhǎng)度不一樣,倆字符串不等, //不必一個(gè)個(gè)比較內(nèi)容,直接返回false }
注:instanceof是Java、php的一個(gè)二元操作符(運(yùn)算符),和==,>,<是同一類東西。由于它是由字母組成的,所以也是Java的保留關(guān)鍵字。它的作用是判斷其左邊對(duì)象是否為其右邊類的實(shí)例,返回boolean類型的數(shù)據(jù)??梢杂脕?lái)判斷繼承中的子類的實(shí)例是否為父類的實(shí)現(xiàn)。
從上面的代碼可以得知:
(1) String類中的equals首先比較地址,如果是同一個(gè)對(duì)象的引用,可知對(duì)象相等,返回true。
(2)如果不是同有一個(gè)對(duì)象,equals方法則繼續(xù)挨個(gè)比較兩個(gè)字符串對(duì)象內(nèi)的字符,只有完全相等才返回true,否則返回false。
三,整理String類的Length()、charAt()、 getChars()、replace()、 toUpperCase()、 toLowerCase()、trim()toCharArray()
使用說(shuō)明
Length():獲取字串長(zhǎng)度
charAt():獲取指定位置的字符
getChars():獲取從指定位置起的子串復(fù)制到字符數(shù)組中
replace():子串替換
toUpperCase()、 toLowerCase():大小寫轉(zhuǎn)換
trim():去除頭尾空格
toCharArray():將字符串對(duì)象轉(zhuǎn)換為字符數(shù)組
四,String類的方法可以連續(xù)調(diào)用:
String str="abc"; String result=str.trim().toUpperCase().concat("defg");
請(qǐng)閱讀JDK中String類上述方法的源碼,模仿其編程方式,編寫一個(gè)MyCounter類,它的方法也支持上述的“級(jí)聯(lián)”調(diào)用特性,其調(diào)用示例為:
MyCounter counter1=new MyCounter(1);
MyCounter counter2=counter1.increase(100).decrease(2).increase(3);
public class MyCounter { int i; MyCounter(int n){ i=n; } public MyCounter increase(int n) { this.i=this.i+n; return this; } public MyCounter decrease(int n) { this.i=this.i-n; return this; } public static void main(String[] args) { MyCounter counter1=new MyCounter(1); MyCounter counter2=counter1.increase(100).decrease(2).increase(3); System.out.println("counter2.i="+counter2.i); } }
總結(jié)
以上就是本文關(guān)于淺談Java編程中string的理解與運(yùn)用的全部?jī)?nèi)容,希望對(duì)大家有所幫助。歡迎參閱:Java實(shí)現(xiàn)微信公眾平臺(tái)朋友圈分享功能詳細(xì)代碼、Java編程BigDecimal用法實(shí)例分享、Java之dao模式詳解及代碼示例等,如有不足之處,歡迎留言指出。感謝朋友們對(duì)本站的支持!
相關(guān)文章
SpringBoot項(xiàng)目打成War包部署的方法步驟
這篇文章主要介紹了springboot項(xiàng)目如何打war包流程的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12Spring boot JPA實(shí)現(xiàn)分頁(yè)和枚舉轉(zhuǎn)換代碼示例
這篇文章主要介紹了Spring boot JPA實(shí)現(xiàn)分頁(yè)和枚舉轉(zhuǎn)換代碼示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-09-09MyBatis Plus之實(shí)現(xiàn)動(dòng)態(tài)排序方式
這篇文章主要介紹了MyBatis Plus之實(shí)現(xiàn)動(dòng)態(tài)排序方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-02-02IDEA 重新導(dǎo)入依賴maven 命令 reimport的方法
這篇文章主要介紹了IDEA 重新導(dǎo)入依賴maven 命令 reimport的相關(guān)知識(shí),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04Springboot?+redis+谷歌開(kāi)源Kaptcha實(shí)現(xiàn)圖片驗(yàn)證碼功能
這篇文章主要介紹了Springboot?+redis+?歌開(kāi)源Kaptcha實(shí)現(xiàn)圖片驗(yàn)證碼功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-01-01Intellij IDEA創(chuàng)建spring-boot項(xiàng)目的圖文教程
本文通過(guò)圖文并茂的形式給大家介紹了Intellij IDEA創(chuàng)建spring-boot項(xiàng)目的教程,本文給大家介紹的非常詳細(xì),具有參考借鑒價(jià)值,需要的朋友參考下吧2018-01-01Windows7下的Java運(yùn)行環(huán)境搭建過(guò)程圖解
這篇文章主要介紹了Windows7下的Java運(yùn)行環(huán)境搭建過(guò)程圖解,需要的朋友可以參考下2014-04-04如何基于springboot-admin實(shí)現(xiàn)后臺(tái)監(jiān)控
這篇文章主要介紹了如何基于springboot-admin實(shí)現(xiàn)后臺(tái)監(jiān)控,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04CAT分布式實(shí)時(shí)監(jiān)控系統(tǒng)使用詳解
這篇文章主要為大家介紹了CAT分布式實(shí)時(shí)監(jiān)控系統(tǒng)介紹詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03