我們在使用IntelliJ的IDE進(jìn)行Debug時,去查看一個未重寫 toString
方法的對象需要展開當(dāng)前的視圖層級才能看到里面的屬性,而通過自定義變量視圖的方式可以直接查看,很大程度上提高Debug的效率。
引出問題
我們在Debug查看變量時通常會遇到這種情況
此時我想查看 Goods
對象里面的具體屬性值,需要點擊左邊的展開按鈕才能查看里面具體的屬性值,不能直接進(jìn)行查看。更糟糕的是當(dāng) Goods
對象在 List
容器中時,我們?nèi)绻肟焖俨檎业疆?dāng)前 List
里元素的某一項(或幾項)屬性時,就會出現(xiàn)在下面的情況,我們只能逐一元素進(jìn)行展開操作才能查看到元素對應(yīng)的信息。
重寫toString及其局限性
上述問題的一般解決方式是重寫該類的 toString
方法,然后重新運(yùn)行Porject,再次Debug時便可以看到變量的視圖會自動變?yōu)?nbsp;toString
方法的返回值,如下
這樣便能無需展開直接顯示 Goods
類的成員變量了,但該方式有幾個缺點
Goods
的 toString
方法之后需要重新運(yùn)行Goods
是被依賴的 jar
這種已經(jīng)被編譯的只讀類,則無法更改Goods
中包含大量屬性(比如20+個)的情況下,無法全部顯示完,所以就無法根據(jù)自己的需求決定查看具體哪些屬性值 自定義變量視圖 IDE提供給我們一種自定義變量視圖的方式,專門用來解決上面的問題并彌補(bǔ)了 toString
方法的不足。這里會有個 變量解析器
的概念,它用來控制當(dāng)前變量的顯示值(即debug時顯示在該變量后面的內(nèi)容,下稱”變量視圖”)。
首先Debug狀態(tài)下右擊變量,選擇 Customize Data Views
項
接下來在 Customize Data Views
彈窗的Tab中選擇 Java Type Renderers
項,如下
點擊 +
來添加一個自定義的 變量解析器
自定義一個 變量解析器
,主要需要添加的是名稱、解析類型和解析方式三部分。
名稱:該解析器的標(biāo)識名稱
解析類型:表示當(dāng)前的解析器只對哪種類型的類進(jìn)行解析
解析方式:此處是核心部分,可以寫一個Java表達(dá)式,也可以寫一段代碼,這里的返回值就是該變量視圖
我們可以在IDE中添加多個 變量解析器
,通過控制它的開啟、禁用、順序、適用類等來控制當(dāng)前變量的顯示情況,下面進(jìn)行一次完整的操作流程演示
通過自定義 變量解析器
的好處是不需要重新運(yùn)行整個Project;而且還可以在Debug期間動態(tài)切換變量視圖,比如
name
屬性時: "name: " + getName()
type
屬性時: "type: " + getType()
title
屬性時: "title: " + getTitle()
再進(jìn)一步抽象
看了官網(wǎng)的自定義Debug變量視圖這部分介紹后,感覺著實對于Debug很實用,通過自實現(xiàn)的方式使得開發(fā)者能夠動態(tài)化控制變量視圖。
自定義 變量解析器
的方式已經(jīng)能夠解決開篇提到的問題,但我更希望能通過它來找到控制變量視圖的通法,即寫一個通用的 變量解析器
而不是每debug一種類型的變量就單獨添加一個解析器。接下來要做的事情很清楚了,就是添加一個能夠?qū)ο髮嵗蛄谢勺址姆椒纯伞W钕认氲降氖峭ㄟ^Json進(jìn)行轉(zhuǎn)化,但Json一般依賴三方包,而我們想讓Debug功能能夠跟IDE是統(tǒng)一基準(zhǔn)線的,所以盡可能選擇使用Jdk自帶的api。
于是考慮到了反射,對于一般通用的變量視圖,我們可以直接通過反射取到每個屬性名,然后結(jié)合當(dāng)前實例來獲取屬性值,直接在上面的 解析類型
中指定為 java.lang.Object
以支持所有類型變量的解析, 解析方式
中添加下面的代碼
if (((Object) this) instanceof String || ((Object) this) instanceof Number || ((Object) this) instanceof Class) { return ((Object) this);}StringBuilder sb = new StringBuilder("{");Class<?> cls = ((Object) this).getClass();java.lang.reflect.Field[] fields = cls.getDeclaredFields();if (fields != null) { int size = fields.length; for (java.lang.reflect.Field field : fields) { field.setAccessible(true); Object value = field.get((Object) this); sb.append(field.getName()) .append("=") .append(String.valueOf(value)); if (--size > 0) { sb.append(", "); } }}return sb.append("}").toString();
添加完畢之后,會發(fā)現(xiàn)此時我們的IDE在Debug時異常強(qiáng)大,所有類型的變量視圖均自動轉(zhuǎn)化成 key-value
形式的字符串,再也不用為了Debug變量而重寫 toString
方法。看下Debug的效果
這里的 User
和 Goods
類均沒有重寫 toString
方法,但都能通過上面添加的通用解析器來進(jìn)行變量視圖解析。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持VEVB武林網(wǎng)。
新聞熱點
疑難解答