詳解Lombok @ToString() 的使用技巧
大家好,我是指北君。
在平時(shí)我們工作的時(shí)候,我們經(jīng)常會(huì)使用toString()? 方法來(lái)輸出一個(gè)對(duì)象的一些屬性信息。Lombok 給我們提供了一個(gè)自動(dòng)生成 toString()?代碼的注解,可以減少代碼行數(shù),如果代碼屬性比較多的話,可以避免我們些代碼的過(guò)程中出現(xiàn)屬性遺漏的問(wèn)題。本文我們來(lái)講講 Lombok 的 ?@ToString()相關(guān)內(nèi)容,以便于我們以后更好的使用 Lombok。
Lombok 的使用
首先我們添加一下 maven 依賴。
然后我們創(chuàng)建一個(gè) Account 的 class 來(lái)演示我們一下我們的各種操作。
默認(rèn)情況下,我們?cè)谡{(diào)用 Account 的toString() 方法之后,輸出的結(jié)果可能如下:
這是一種比較標(biāo)準(zhǔn)的格式輸出。
Lombok的配置
父類 toString() 的調(diào)用
現(xiàn)在假設(shè)我們有一個(gè) SavingAccount 的 class 繼承于 Account ,我們調(diào)用 SavingAccount 的 toString() 方法時(shí),希望把 Account 的一些屬性也能夠一起輸出, 這個(gè)時(shí)候我們可以設(shè)置 callSupper 屬性來(lái)達(dá)到我們的目的。
上述代碼的操作,就能把父類 Account 的屬性都輸出出來(lái):
省略字段名稱
我們知道默認(rèn)輸出的時(shí)候,會(huì)包含字段名稱,我們可以通過(guò)設(shè)置 includeFieldNames 來(lái)控制,是否顯示屬性名稱。
把 includeFieldNames 設(shè)置為 false 之后,輸出結(jié)果如下
使用字段代替 Getter
我們知道 getter 方法提供了用于打印的字段值。如果該類不包含某個(gè)特定字段的getter方法,那么Lombok會(huì)直接訪問(wèn)該字段并獲取其值。我們可以通過(guò)設(shè)置 ?doNotUseGetters 屬性為 true,將 Lombok 配置為總是使用直接的字段值而不是getter。
如果沒(méi)有這個(gè)屬性,我們會(huì)得到通過(guò)調(diào)用getters得到的輸出。
相反,通過(guò)設(shè)置doNotUseGetters屬性,輸出實(shí)際上顯示了id字段的值,而沒(méi)有調(diào)用getter。
字段的包含和排除
假設(shè)我們想從字符串表示中排除某些字段,例如,密碼、其他敏感信息或大的JSON結(jié)構(gòu)。我們可以通過(guò)@ToString.Exclude注解來(lái)省略這些字段。讓我們把名字字段從我們的表示中排除
或者,我們可以只指定輸出中所需的字段,我可以通過(guò)使用 @ToString(onlyExplicitlyIncluded = true) ?和 @ToString.Include來(lái)實(shí)現(xiàn)。
上述兩種方法,最終輸出,都只能輸出 id 字段。
另外,Lombok 會(huì)自動(dòng)忽略以$ 開(kāi)頭的變量,但是我們可以通過(guò) @ToString.Include 來(lái)強(qiáng)制Lombok輸出。
輸出排序
默認(rèn)情況下,Lombok 的輸出,是按照字段定義的順序進(jìn)行輸出的,我可以通過(guò)設(shè)置 @ToString.Include 來(lái)進(jìn)行排序。我們先修改一下 Account 的字段順序, 然后對(duì) id 進(jìn)行標(biāo)記順序。
現(xiàn)在 id 字段輸出的時(shí)候,會(huì)排在 name 的前面
Lombok 輸出的規(guī)則大致如下:
- rank 排名越大,排序越靠前
- 默認(rèn)的排序值為0
- 相同的排序通過(guò)根據(jù)字段定義順序輸出
方法輸出
除了字段之外,我們也可以包括一個(gè)不需要參數(shù)的實(shí)例方法的輸出。我們可以通過(guò)用@ToString.Include標(biāo)記無(wú)參數(shù)的實(shí)例方法來(lái)做到這一點(diǎn)。
這里 description 將會(huì)作為輸出 key 進(jìn)行打印輸出。
如果指定的方法名稱與字段名稱相匹配,那么該方法就會(huì)優(yōu)先于字段。換句話說(shuō),輸出包含方法調(diào)用的結(jié)果,而不是匹配字段的值。
修改字段名稱
我們可以通過(guò) @ToString.Include 的屬性來(lái)修改字段的名稱。
現(xiàn)在輸出結(jié)果中,將不會(huì)包含字段名稱id ,將會(huì)輸出 identification。
打印數(shù)組
Lombok 使用 Arrays.deepToString() 方法打印數(shù)組,將數(shù)組元素轉(zhuǎn)換為其相應(yīng)的字符串表示。但是數(shù)組有可能包含直接引用或間接循環(huán)引用。為了避免無(wú)限遞歸及其相關(guān)的運(yùn)行時(shí)錯(cuò)誤,該方法將任何從自身內(nèi)部對(duì)數(shù)組的循環(huán)引用渲染為"``.``.``.``"。讓我們通過(guò)給我們的賬戶類添加一個(gè)對(duì)象數(shù)組字段來(lái)看看。
這 relatedAccounts 數(shù)組的打印如下
重要的是,循環(huán)引用被deepToString()方法檢測(cè)到,并且被Lombok適當(dāng)?shù)爻尸F(xiàn)出來(lái),沒(méi)有引起任何StackOverflowError。
有一些注意點(diǎn)
有幾個(gè)細(xì)節(jié)值得一提,對(duì)避免產(chǎn)生意外的結(jié)果很重要。
- 在類中存在任何名為toString()的方法(不管返回類型如何),Lombok不會(huì)生成其 toString() 方法。
- 不同版本的Lombok可能會(huì)改變生成方法的輸出格式。在任何情況下,我們應(yīng)該避免依賴解析toString()方法輸出的代碼。所以這其實(shí)不應(yīng)該是一個(gè)問(wèn)題。
- 我們還可以在枚舉上添加這個(gè)注解。這將產(chǎn)生一個(gè)枚舉值跟隨枚舉類名稱的表示,例如,AccounType.SAVING。