您好,登錄后才能下訂單哦!
這篇文章主要介紹JAVA中字符串和數組做參數傳遞的示例分析,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!
首先明確的一點就是在java中只有值傳遞!只有值傳遞!理論依據來自《think in java》。接下來就是具體說明為何java只有值傳遞。
因為java中有基本類型和引用類型兩種數據類型,再加上String這個特殊的類型,所以主要從三個方面就行解釋。
先看代碼
public class Demo01 { public void change(int a) { System.out.println("副本a 的初始值" + a); a = 20; System.out.println("副本a 的新值值" + a); } public static void main(String[] args) { int a = 10; Demo01 d = new Demo01(); d.change(a); System.out.println("change方法執行后的值" + a); } }
分析:
在java中基本數據類型遵循值傳遞,所以對象d在調用change()方法時,只是將原數據a的副本傳給方法中的參數,第一時間原本和副本a的值都是10,在執行到a=20后,副本a的值變成了20。
所以運行結果為:
原理參考下圖
先看代碼
public class Demo02 { char[] ch = {'a', 'b', 'c'}; public void change(char ch[]) { System.out.println("方法中ch[0]的初始值:" + ch[0]); ch[0] = 'g'; System.out.println("方法中ch[0]執行后的新值:" + ch[0]); } public static void main(String[] args) { Demo02 d = new Demo02(); System.out.println("對象d中數組的初始值是:"+d.ch); d.change(d.ch); System.out.println("對象d中數組的最終值是:"+d.ch); } }
分析:
在引用類型作為參數進行傳遞時,也屬于值傳遞,此時傳遞的是地址值副本,但是這兩個地址指向同一個地方。在副本地址沒有進行更改指向時,對副本地址指向的數據進行操作會影響到原始數據的值。方法中ch[] 數組和原始ch[]數組指向同一個數據,所以初始階段ch[0]都指向’a’;接著對副本中的ch[0]進行新的賦值變為‘g’。
所以運行結果為:
原理參考下圖
先看代碼
public class Demo03 { public void change(String str2) { System.out.println("方法中str2初始值" + str2); System.out.println("方法中str2初始hashcode值" + str2.hashCode()); str2 = "bbb"; System.out.println("方法中str2賦值后:" + str2); System.out.println("方法中str2賦值后hashcode值:" + str2.hashCode()); } public static void main(String[] args) { String str1 = new String("aaa"); System.out.println("原始字符串str1的hashcode值:" + str1.hashCode()); Demo03 d = new Demo03(); d.change(str1); System.out.println("方法調用后str1的值" + str1); } }
分析:
字符串是一個特殊的數據類型,它的底層是一個final 型的char[]數組,屬于無法更改,所以字符串在作為參數傳遞時,可以當做一個特殊的數組進行操作,同樣的它也是將復制一份原本的對象引用給了副本,此時副本對象的引用和原本對象的引用都指向原始字符串的位置,也就是str2在剛開始初始化時它指向的地址和原對象str1指向的位置一致,即str2的初始hashcode值和原對象str1的hashcode值一樣,str2經過str2=“bbb”操作后,由于字符串的不可變性,此時str2會指向一個新的對象引用,即此時str2指向“bbb”的位置。str2的hashcode值會變化,但是原本str1它的對象引用沒有發生改變,并且“aaa”也未發生改變,所以str1仍然指向”aaa”。運行結果如下:
接下來看一個更具體的字符串例子:
public class Demo04 { public static void main(String[] args) { StringBuffer s = new StringBuffer("hello"); StringBuffer s2 = new StringBuffer("hi"); test(s, s2); System.out.println("方法調用后s的值:" + s); System.out.println("方法調用后s2的值:" + s2); } static void test(StringBuffer s3, StringBuffer s4) { System.out.println("方法初始化時s3的值" + s3); System.out.println("方法初始化時s4的值" + s4); s4 = s3; s3 = new StringBuffer("new"); System.out.println("第一步變化后s3的值" + s3); System.out.println("第一步變化后s4的值" + s4); s3.append("boy"); s4.append("gril"); System.out.println("第二步變化后s3的值" + s3); System.out.println("第二步變化后s4的值" + s4); } }
這次先看結果:
然后進行分析:
在未執行方法之前,字符串s1和s2指向的位置分別是“hello”和“hi”,這個毋容置疑,
(1)接著進入方法內部,方法中參數s3和s4初始化時和上面例子相同,此時它們和s1s2指向同一個位置,或者說s1s2將對象引用副本給了s3s4,此時s3s4的值為“hello”和“hi”
(2)接著執行s4=s3,這個操作就是將s3的對象引用給了s4,此時s4為“hello”;s3=new StringBuffer(”new”);這個操作要注意,此時相當于給了s3一個新的對象引用,s3指向一個字符串為“new”的位置,所以此時s3=“new”,s4=“hello”
(3)然后s3.append(“boy”);s4.append(“gril”);在StringBuffer中的append方法要注意,它的操作不會為s3s4指向一個新的對象引用,是在原來的基礎上進行操作,因此操作完之后s3=“newboy”,s4=“hellogril”
(4)此時方法調用完,回頭捋一下s3s4在此過程中的對s1s2的影響。
——- A . 首先是s3和s1一樣剛開始指向“hello”,接著給s3創建一個新的對象引用“new”,此時s3和s1再無半毛錢關系,s3進行append(boy)后,s3=“newboy”;
——– B . s4剛開始和s2都指向“hi”,接著s3將自己初始值(也就是s1的副本)給了s4,此時s4指向“hello”(這會s4和s1有了關系),s4執行append(grill)操作,因為它和s1指向相同位置,所以它們的共同指向的對象會變化,s4=s1=“hellogrill”。
——- C .然后就清楚了,s2指向的對象“hi”并未變化,s1指向的“hello”在append(“grill”)操作下變成了“hellogril”。
當使用基本數據類型作為方法的形參時,在方法體中對形參的修改不會影響到實參的數值
當使用引用數據類型作為方法的形參時,若在方法體中修改形參指向的數據內容,會對實參變量的數值產生影響,因為形參變量和實參變量共享同一塊堆區;
當使用引用數據類型作為方法的形參時,若在方法體中改變了形參變量的指向,此時不會對實參變量的數值產生影響,因此形參變量和實參變量分別指向不同的堆區;最后一個例子就是最形象的解釋。
關于字符串做參數,也是看它的參數變量指向是否發生了變化,因為String的底層為final類型的char[]原因,當你在String s = “aaa”還是String s = new String(“aaa”)時,都會為s創建一個新的對象引用。但是調用了append()方法時,是不會指向新的對象,會在原來的指向的對象上發生改變,與它共享的對象引用也會發生變化。
最后重復的是java中沒有引用傳遞,只有值傳遞,引用類型屬于特殊值傳遞(是將它的地址副本給了參數,但是它與基本數據類型不同,如果地址指向的對象發生了變化,因為共享原因,原始對象也會改變)。
以上是“JAVA中字符串和數組做參數傳遞的示例分析”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。