妹:打印數(shù)組有什么好講的?
“哥,之前聽(tīng)你說(shuō),數(shù)組也是一個(gè)對(duì)象,但 Java 中并未明確的定義這樣一個(gè)類(lèi)。”看來(lái)三妹有在用心地學(xué)習(xí)。
“是的,因此數(shù)組也就沒(méi)有機(jī)會(huì)覆蓋 Object.toString() 方法。如果嘗試直接打印數(shù)組的話,輸出的結(jié)果并不是我們預(yù)期的結(jié)果。”我接著三妹的話繼續(xù)說(shuō)。
“那怎么打印數(shù)組呢?”三妹心有靈犀地把今天的核心問(wèn)題提了出來(lái)。
“首先,我們來(lái)看一下,為什么不能直接打印數(shù)組,直接打印的話,會(huì)出現(xiàn)什么問(wèn)題。”
來(lái)看這樣一個(gè)例子。
- String [] cmowers = {"沉默","王二","一枚有趣的程序員"};
- System.out.println(cmowers);
程序打印的結(jié)果是:
- [Ljava.lang.String;@3d075dc0
[Ljava.lang.String; 表示字符串?dāng)?shù)組的 Class 名,@ 后面的是十六進(jìn)制的 hashCode——這樣的打印結(jié)果太“人性化”了,一般人表示看不懂!為什么會(huì)這樣顯示呢?查看一下 java.lang.Object 類(lèi)的 toString() 方法就明白了。
- public String toString() {
- return getClass().getName() + "@" + Integer.toHexString(hashCode());
- }
再次證明,數(shù)組雖然沒(méi)有顯式定義成一個(gè)類(lèi),但它的確是一個(gè)對(duì)象,繼承了祖先類(lèi) Object 的所有方法。
“哥,那為什么數(shù)組不單獨(dú)定義一個(gè)類(lèi)來(lái)表示呢?就像字符串 String 類(lèi)那樣呢?”三妹這個(gè)問(wèn)題讓人頭大,但也好解釋。
“一個(gè)合理的說(shuō)法是 Java 將其隱藏了。假如真的存在這么一個(gè)類(lèi),就叫 Array.java 吧,我們假想一下它真實(shí)的樣子,必須得有一個(gè)容器來(lái)存放數(shù)組的每一個(gè)元素,就像 String 類(lèi)那樣。”一邊回答三妹,我一邊打開(kāi)了 String 類(lèi)的源碼。
- public final class String
- implements java.io.Serializable, Comparable<String>, CharSequence {
- /** The value is used for character storage. */
- private final char value[];
- }
“最終還是要用類(lèi)似一種數(shù)組的形式來(lái)存放數(shù)組的元素,對(duì)吧?這就變得很沒(méi)有必要了,不妨就把數(shù)組當(dāng)做是一個(gè)沒(méi)有形體的對(duì)象吧!”
“好了,不討論這個(gè)了。”我怕話題扯遠(yuǎn)了,扯到我自己也答不出來(lái)就尷尬了,趕緊把三妹的思路拽了回來(lái)。
“我們來(lái)看第一種打印數(shù)組的方法,使用時(shí)髦一點(diǎn)的 Stream 流。”
第一種形式:
- Arrays.asList(cmowers).stream().forEach(s -> System.out.println(s));
第二種形式:
- Stream.of(cmowers).forEach(System.out::println);
第三種形式:
- Arrays.stream(cmowers).forEach(System.out::println);
打印的結(jié)果如下所示。
- 沉默
- 王二
- 一枚有趣的程序員
沒(méi)錯(cuò),這三種方式都可以輕松勝任本職工作,并且顯得有點(diǎn)高大上,畢竟用到了 Stream,以及 lambda 表達(dá)式。
“當(dāng)然了,也可以使用比較土的方式,for 循環(huán)。甚至 for-each 也行。”
- for(int i = 0; i < cmowers.length; i++){
- System.out.println(cmowers[i]);
- }
- for (String s : cmowers) {
- System.out.println(s);
- }
“哥,你難道忘了上一篇在講 Arrays 工具類(lèi)的時(shí)候,提到過(guò)另外一種方法 Arrays.toString() 嗎?”三妹看我一直說(shuō)不到點(diǎn)子上,有點(diǎn)著急了。
“當(dāng)然沒(méi)有了,我認(rèn)為 Arrays.toString() 是打印數(shù)組的最佳方式,沒(méi)有之一。”我的情緒有點(diǎn)激動(dòng)。
Arrays.toString() 可以將任意類(lèi)型的數(shù)組轉(zhuǎn)成字符串,包括基本類(lèi)型數(shù)組和引用類(lèi)型數(shù)組。該方法有多種重載形式。
使用 Arrays.toString() 方法來(lái)打印數(shù)組再優(yōu)雅不過(guò)了,就像,就像,就像蒙娜麗莎的微笑。
(三妹看到這么一副圖的時(shí)候忍不住地笑了)
“三妹,你不要笑,來(lái),懷揣著愉快的心情看一下代碼示例。”
- String [] cmowers = {"沉默","王二","一枚有趣的程序員"};
- System.out.println(Arrays.toString(cmowers));
程序打印結(jié)果:
- [沉默, 王二, 一枚有趣的程序員]
哇,打印格式不要太完美,不多不少!完全是我們預(yù)期的結(jié)果:[] 表明是一個(gè)數(shù)組,, 點(diǎn)和空格用來(lái)分割元素。
“哥,那如果我想打印二維數(shù)組呢?”
“可以使用 Arrays.deepToString() 方法。”
- String[][] deepArray = new String[][] {{"沉默", "王二"}, {"一枚有趣的程序員"}};
- System.out.println(Arrays.deepToString(deepArray));
打印結(jié)果如下所示。
- [[沉默, 王二], [一枚有趣的程序員]]
“說(shuō)到打印,三妹,哥給你提醒一點(diǎn)。阿里巴巴的 Java 開(kāi)發(fā)手冊(cè)上有這樣一條規(guī)約,你看。”
“什么是 POJO 呢,就是 Plain Ordinary Java Object 的縮寫(xiě),一般在 Web 應(yīng)用程序中建立一個(gè)數(shù)據(jù)庫(kù)的映射對(duì)象時(shí),我們稱(chēng)它為 POJO,這類(lèi)對(duì)象不繼承或不實(shí)現(xiàn)任何其它 Java 框架的類(lèi)或接口。”
“對(duì)于這樣的類(lèi),最好是重寫(xiě)一下它的 toString() 方法,方便查看這個(gè)對(duì)象到底包含了什么字段,好排查問(wèn)題。”
“如果不重寫(xiě)的話,打印出來(lái)的 Java 對(duì)象就像直接打印數(shù)組的那樣,一串誰(shuí)也看不懂的字符序列。”
“可以借助 Intellij IDEA 生成重寫(xiě)的 toString() 方法,特別方便。”
“好的,哥,我記住了。以后遇到的話,我注意下。你去休息吧,我來(lái)敲一下你提到的這些代碼,練一練。”
“OK,我走,我走。”
本文轉(zhuǎn)載自微信公眾號(hào)「沉默王二」,可以通過(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系沉默王二公眾號(hào)。