0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線(xiàn)課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

重演自己如何掉入Lombok的戲法陷阱

Android編程精選 ? 來(lái)源:ramostear ? 作者:樹(shù)下魅狐 ? 2021-10-28 11:29 ? 次閱讀
https://www.ramostear.com/blog/2020/04/28/uk1860p8.html 如果您正在閱讀此文,想必您對(duì)Project Lombok已經(jīng)有了一段時(shí)間的了解。您是否正準(zhǔn)備擁抱Lombok?還是正準(zhǔn)備將如此酷炫的項(xiàng)目推薦給你的團(tuán)隊(duì)?如果您準(zhǔn)備那么做,不妨聽(tīng)聽(tīng)我在使用Lombok一年后的一些感受。我承認(rèn),Lombok是一個(gè)很不錯(cuò)的Java庫(kù),它可以讓你在少寫(xiě)代碼的同時(shí)耍???,簡(jiǎn)單的幾個(gè)注解,就可以干掉一大片模板代碼。但是,所有的源代碼很多時(shí)候是用來(lái)閱讀的,只有很少的時(shí)間是用來(lái)執(zhí)行的(你可以細(xì)品這句話(huà))。 一年以前,我和大多數(shù)人都認(rèn)為L(zhǎng)ombok的出現(xiàn)會(huì)讓Java的編碼體驗(yàn)會(huì)更好,并極力的在我的團(tuán)隊(duì)中推薦使用Lombok。一年以后,我開(kāi)始對(duì)此產(chǎn)生顧慮,尤其是在我準(zhǔn)備為開(kāi)源的博客系統(tǒng)Una-Boot升級(jí)Java版本時(shí),我才意識(shí)到Lombok自己掉入了一個(gè)戲法陷阱。在我進(jìn)一步分析其源代碼并理解相關(guān)注解的工作原理后,發(fā)現(xiàn)我并不需要使用一個(gè)非標(biāo)準(zhǔn)的第三方庫(kù)將Java轉(zhuǎn)換為一個(gè)精巧而酷炫的語(yǔ)言。引入Lombok讓我的項(xiàng)目一時(shí)爽,但一時(shí)爽的代價(jià)是隨著項(xiàng)目推進(jìn),技術(shù)債務(wù)開(kāi)始累積。接下來(lái),我將用幾個(gè)大家耳熟能詳?shù)膱?chǎng)景,重演自己是如何掉入Lombok的戲法陷阱。

愛(ài)的開(kāi)始,恨的起源

面對(duì)Lombok提供的諸多“神走位”,你并不會(huì)介意在IDE上新增一個(gè)插件。對(duì)于IntelliJ IDEA玩家而言,只需搜索“Lombok Plugin”便可找到這款神器并安裝上它。愛(ài)上Lombok從安裝Lombok插件開(kāi)始,恨也從此萌芽。沒(méi)使用Lombok之前,我們的源代碼看起來(lái)是這一的:

	

publicclassMyObject{ privateLongid; privateStringname; privateintage; privateintgender; publicLonggetId(){ returnid; } publicvoidsetId(Longid){ this.id=id; } publicStringgetName(){ returnname; } publicvoidsetName(Stringname){ this.name=name; } publicintgetAge(){ returnage; } publicvoidsetAge(intage){ this.age=age; } publicintgetGender(){ returngender; } publicvoidsetGender(intgender){ this.gender=gender; } @Override publicbooleanequals(Objecto){ if(this==o){ returntrue; } if(o==null||getClass()!=o.getClass()){ returnfalse; } MyObjectobj=(MyObject)o; returnage=obj.age&& gender=obj.gender&& Objects.equals(id,obj.id)&& Objects.queals(name,obj.name); } @Override publicinthashCode(){ returnObjects.hash(id,name,age,gender); } @Override publicStringtoString(){ return"MyObject{"+ "id="+id+ "name="+name+ "age="+age+ "gender="+gander+ "}"; } }

每個(gè)JavaBean都會(huì)充斥著如上述getter,setter,equals,hashCode和toString的模板代碼,這看起來(lái)像一個(gè)偏胖的人(不得不承認(rèn)Java是一個(gè)有缺陷的編程語(yǔ)言)。當(dāng)我們安裝好Lombok插件后,IDE便可以識(shí)別其酷炫的注解,使用Lombok的@Getter@Setter注解后,代碼會(huì)像下面這樣看起來(lái)很苗條:

	

@Getter @Setter publicclassMyObject{ privateLongid; privateStringname; privateintage; privateintgender; @Override publicbooleanequals(Objecto){ if(this==o){ returntrue; } if(o==null||getClass()!=o.getClass()){ returnfalse; } MyObjectobj=(MyObject)o; returnage=obj.age&& gender=obj.gender&& Objects.equals(id,obj.id)&& Objects.queals(name,obj.name); } @Override publicinthashCode(){ returnObjects.hash(id,name,age,gender); } @Override publicStringtoString(){ return"MyObject{"+ "id="+id+ "name="+name+ "age="+age+ "gender="+gander+ "}"; } }

你以為L(zhǎng)ombok就這點(diǎn)能耐?它還能讓你代碼的“身材”更苗條,更魔鬼。上面的代碼仍然還有改進(jìn)的空間,我們可以用@EqualsAndHashCod注解替換到equals和hashCode方法:

	

@Getter @Setter @EqualsAndHashCode publicclassMyObject{ privateLongid; privateStringname; privateintage; privateintgender; @Override publicStringtoString(){ return"MyObject{"+ "id="+id+ "name="+name+ "age="+age+ "gender="+gander+ "}"; } }

現(xiàn)在的代碼是否看起來(lái)爽多了?但這還不是最爽的時(shí)候。既然其他方法都替換掉了,那把toString方法也一起拿掉吧.如你所愿,可以使用@ToString注解去掉對(duì)于的方法:

	

@Getter @Setter @EqualsAndHashCode @ToString publicclassMyObject{ privateLongid; privateStringname; privateintage; privateintgender; }

經(jīng)過(guò)Lombok的戲法之后,相比一開(kāi)始的代碼,看起來(lái)是不是很酷炫,很苗條,很性感?你以為到此為止了?遠(yuǎn)不止于此。你會(huì)發(fā)現(xiàn)類(lèi)名上一大坨注解看起來(lái)好別扭,Lombok提供了一個(gè)組合注解@Data,可以替換掉類(lèi)名頭上那坨像翔一樣的東西:

	

@Data publicclassMyObject{ privateLongid; privateStringname; privateintage; privateintgender; }

現(xiàn)在,Lombok是否讓你的對(duì)象成為了你心目中完美的樣子?魔鬼的“身材”,酷炫精煉。Lombok還有其他一些注解,如@Slf4j,@NoArgsConstructor@AllArgsConstructor等等,介紹Lombok用法不是本文重點(diǎn)。 以上代碼行數(shù)的變化過(guò)程,也許是無(wú)數(shù)程序員愛(ài)上Lombok的主要原因吧,這就像一個(gè)肥胖的人逐漸變成一個(gè)身材苗條的人。同時(shí)也讓你看到了一個(gè)現(xiàn)象:你以為程序員很懶嗎?其他有些時(shí)候他們比你想象中的還要懶。在爽的同時(shí),也為代碼種下了禍根

扭曲的審美,愛(ài)的隱患

扭曲的審美,導(dǎo)致了被審視的對(duì)象出于亞健康狀態(tài)。使用Lombok插件之后,我們的代碼也處于“亞健康”狀態(tài)。還是回歸一開(kāi)始的那句話(huà):所有的源代碼很多時(shí)候是用來(lái)閱讀的,只有很少的時(shí)間是用來(lái)執(zhí)行的。 本質(zhì)上講,我們都追求減少程序中的樣板代碼以使其代碼更精煉簡(jiǎn)潔,從而提高代碼的可讀性和可維護(hù)性。但Lombok并沒(méi)有達(dá)到我們所追求的這一愿景,它僅僅是利用Java語(yǔ)言在編譯時(shí)的空檔期,使用一種很取巧的方式,將我們所需要的方法注入(寫(xiě)入)到當(dāng)前的類(lèi)中,這種過(guò)程很像在hack我們的代碼,只是一種看起來(lái)酷炫的把戲。這種把戲并不智能和安全,反而會(huì)破壞Java代碼現(xiàn)有的特性以及代碼的可讀性。下面,結(jié)合我自己使用Lombok之后的感受,談?wù)凩ombok帶來(lái)的幾大痛點(diǎn)。

1. JDK版本問(wèn)題

當(dāng)我想要將現(xiàn)有項(xiàng)目的JDK從Java 8升級(jí)到Java 11時(shí),我發(fā)現(xiàn)Lombok不能正常工作了。于是我不得不將所有的Lombok注解從項(xiàng)目源代碼中清除,并使用IDE自帶的功能生成getter/setter,equals,hashCode,toString以及構(gòu)造器等方法,你也可以使用Delombok工具完成這一過(guò)程。但這終究會(huì)消耗你很多的時(shí)間。

2. 脅迫使用

當(dāng)你的源代碼中使用了Lombok,恰好你的代碼又被其他的人所使用,那么依賴(lài)你代碼的人,也必須安裝Lombok插件(不管他們喜不喜歡),同時(shí)還要花費(fèi)時(shí)間去了解Lombok注解的使用情況,如果不那么做,代碼將無(wú)法正常運(yùn)行。使用過(guò)Lombok之后,我發(fā)現(xiàn)這是一種很流氓的行為。

3. 可讀性差

Lombok隱藏了JavaBean封裝的細(xì)節(jié),如果你使用@AllArgsConstructor注解,它將提供一個(gè)巨型構(gòu)造器,讓外界有機(jī)會(huì)在初始化對(duì)象時(shí)修改類(lèi)中所有的屬性。首先,這是極其不安全的,因?yàn)轭?lèi)中某系屬性我們是不希望被修改的;另外,如果某個(gè)類(lèi)中有幾十個(gè)屬性存在,就會(huì)有一個(gè)包含幾十個(gè)參數(shù)的構(gòu)造器被Lombok注入到類(lèi)中,這是不理智的行為;其次,構(gòu)造器參數(shù)的順序完全由Lombok所控制,我們并不能操控,只有當(dāng)你需要調(diào)試時(shí)才發(fā)現(xiàn)有一個(gè)奇怪的“小強(qiáng)”在等著你;最后,在運(yùn)行代碼之前,所有JavaBean中的方法你只能想象他們長(zhǎng)什么樣子,你并不能看見(jiàn)。

4. 代碼耦合度增加

當(dāng)你使用Lombok來(lái)編寫(xiě)某一個(gè)模塊的代碼后,其余依賴(lài)此模塊的其他代碼都需要引入Lombok依賴(lài),同時(shí)還需要在IDE中安裝Lombok的插件。雖然Lombok的依賴(lài)包并不大,但就因?yàn)槠渲幸粋€(gè)地方使用了Lombok,其余所有的依賴(lài)方都要強(qiáng)制加入Lombok的Jar包,這是一種入侵式的耦合,如果再遇上JDK版本問(wèn)題,這將是一場(chǎng)災(zāi)難。

5. 得不償失

使用Lombok,一時(shí)覺(jué)得很爽,但它卻污染了你的代碼,破壞了Java代碼的完整性,可讀性和安全性,同時(shí)還增加的團(tuán)隊(duì)的技術(shù)債務(wù),這是一種弊大于利,得不償失的操作。如果你確實(shí)想讓自己的代碼更加精煉,同時(shí)又兼顧可讀性和編碼效率,不妨使用主流的Scala或Kotlin這一基于JVM的語(yǔ)言。

總結(jié)

Lombok本身是一個(gè)優(yōu)秀的Java代碼庫(kù),它采用了一種取巧的語(yǔ)法糖,簡(jiǎn)化了Java的編碼,為Java代碼的精簡(jiǎn)提供了一種方式,但在使用此代碼庫(kù)時(shí),需要了解到Lombok并非一個(gè)標(biāo)準(zhǔn)的Java庫(kù)。使用Lombok,會(huì)增加團(tuán)隊(duì)的技術(shù)債務(wù),降低代碼的可讀性,增大代碼的耦合度和調(diào)式難度。雖然在一定程度上Lombok減少了樣板代碼的書(shū)寫(xiě),但也帶來(lái)了一些未知的風(fēng)險(xiǎn)。如果你正在參與一個(gè)團(tuán)隊(duì)項(xiàng)目(或大型項(xiàng)目),考慮到后續(xù)的升級(jí)與擴(kuò)展,是否使用Lombok,請(qǐng)與你的團(tuán)隊(duì)多溝通和三思。
責(zé)任編輯:haq

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • JAVA
    +關(guān)注

    關(guān)注

    19

    文章

    2943

    瀏覽量

    104096
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4671

    瀏覽量

    67765

原文標(biāo)題:Lombok!代碼簡(jiǎn)潔神器還是代碼“亞健康”元兇?

文章出處:【微信號(hào):AndroidPush,微信公眾號(hào):Android編程精選】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    PyTorch如何訓(xùn)練自己的數(shù)據(jù)集

    PyTorch是一個(gè)廣泛使用的深度學(xué)習(xí)框架,它以其靈活性、易用性和強(qiáng)大的動(dòng)態(tài)圖特性而聞名。在訓(xùn)練深度學(xué)習(xí)模型時(shí),數(shù)據(jù)集是不可或缺的組成部分。然而,很多時(shí)候,我們可能需要使用自己的數(shù)據(jù)集而不是現(xiàn)成
    的頭像 發(fā)表于 07-02 14:09 ?540次閱讀

    Lombok注解引發(fā)的空指針問(wèn)題分析

    一、問(wèn)題描述 在一次上線(xiàn)后,日志中出現(xiàn)空指針的報(bào)錯(cuò),但是報(bào)錯(cuò)代碼位置以及相應(yīng)工具類(lèi)未進(jìn)行過(guò)修改,接下來(lái)進(jìn)一步分析。 以下為報(bào)錯(cuò)堆棧信息: ? java.lang.NullPointerException: nullat net.sf.cglib.core.ReflectUtils.getMethodInfo(ReflectUtils.java:424) ~[cglib-3.1.jar:?]at net.sf.cglib.beans.BeanCopier$Generator.generateClass(BeanCopier.java:133) ~[cglib-3.1.jar:?]at net.sf.cglib.core.DefaultGeneratorStrategy.generate(DefaultGeneratorStrategy.java:25) ~[cglib-3.1.jar:?]at net.sf.cglib.core.AbstractClassGenerator.create(AbstractClassGene
    的頭像 發(fā)表于 06-23 09:30 ?926次閱讀
    <b class='flag-5'>Lombok</b>注解引發(fā)的空指針問(wèn)題分析

    如何選擇合適自己的光纖涂覆機(jī)

    電子發(fā)燒友網(wǎng)站提供《如何選擇合適自己的光纖涂覆機(jī).pdf》資料免費(fèi)下載
    發(fā)表于 06-06 11:37 ?0次下載

    如何制作自己的Arduino電容計(jì)

    在這個(gè)項(xiàng)目中,您將學(xué)習(xí)如何制作自己的Arduino電容計(jì)(測(cè)量電容器的值,范圍從pF到1000的uF)。一般來(lái)說(shuō),電子愛(ài)好者喜歡設(shè)計(jì)自己的小工具而不是購(gòu)買(mǎi)。在這個(gè)項(xiàng)目中,我們使用兩種電容測(cè)量方法,即
    的頭像 發(fā)表于 02-25 15:10 ?1150次閱讀
    如何制作<b class='flag-5'>自己</b>的Arduino電容計(jì)

    開(kāi)局2024:碳酸鋰庫(kù)存重回7萬(wàn)噸“背后” 節(jié)后反彈,希望幾何?

    正極開(kāi)啟補(bǔ)庫(kù),節(jié)后鋰價(jià)反彈或“重演”。
    的頭像 發(fā)表于 01-26 09:53 ?926次閱讀

    如何自己搭建一個(gè)服務(wù)器?

    如何自己搭建一個(gè)服務(wù)器?自己搭建一個(gè)服務(wù)器涉及到硬件和軟件兩個(gè)方面。下面是一個(gè)簡(jiǎn)單的指南,涵蓋了基本的步驟。請(qǐng)注意,這是一個(gè)概述,實(shí)際步驟可能因你的需求和硬件選擇而有所不同。
    的頭像 發(fā)表于 12-12 16:52 ?3538次閱讀

    華為星閃BS25 SDK中GADC設(shè)置的陷阱

    用華為星閃 BS25 SDK 做產(chǎn)品中發(fā)現(xiàn) GADC中的 一個(gè)小陷阱。
    的頭像 發(fā)表于 12-01 14:01 ?1414次閱讀
    華為星閃BS25 SDK中GADC設(shè)置的<b class='flag-5'>陷阱</b>

    如何避開(kāi)無(wú)源元件的陷阱

    電子發(fā)燒友網(wǎng)站提供《如何避開(kāi)無(wú)源元件的陷阱.pdf》資料免費(fèi)下載
    發(fā)表于 11-28 10:19 ?0次下載
    如何避開(kāi)無(wú)源元件的<b class='flag-5'>陷阱</b>

    python如何import自己的模塊

    在Python中,我們可以通過(guò)import語(yǔ)句來(lái)引入自己的模塊。導(dǎo)入自己的模塊可以提高代碼的可重用性,使代碼更加清晰和易于維護(hù)。在本文章中,我們將詳細(xì)介紹如何導(dǎo)入自己的模塊以及一些附加的相關(guān)主題
    的頭像 發(fā)表于 11-22 14:57 ?1931次閱讀

    Python項(xiàng)目中遇到的10個(gè)安全陷阱

    Python 開(kāi)發(fā)者們?cè)谑褂脴?biāo)準(zhǔn)庫(kù)和通用框架時(shí),都以為自己的程序具有可靠的安全性。然而,在 Python 中,就像在任何其它編程語(yǔ)言中一樣,有一些特性可能會(huì)被開(kāi)發(fā)者們誤解或誤用。通常而言,只有
    的頭像 發(fā)表于 10-30 10:34 ?390次閱讀

    Lombok如何派上用場(chǎng)

    Java的 record 關(guān)鍵字是Java 14中引入的一個(gè)新的語(yǔ)義特性。record 對(duì)于創(chuàng)建小型不可變的對(duì)象非常有用。另一方面,Lombok 是一個(gè)Java庫(kù),可以自動(dòng)生成一些已知的模式為
    的頭像 發(fā)表于 10-08 15:39 ?257次閱讀

    如何編寫(xiě)一個(gè)自己的 starter

    前面給大家介紹了 SpringBoot 的自動(dòng)裝配功能,相信大家對(duì)自動(dòng)裝配都有了很好的理解,那么今天阿粉通過(guò)一個(gè)示例來(lái)給大家演示一下如何編寫(xiě)一個(gè)自己的 starter 。 再編寫(xiě) starter
    的頭像 發(fā)表于 10-08 14:42 ?753次閱讀
    如何編寫(xiě)一個(gè)<b class='flag-5'>自己</b>的 starter

    U54內(nèi)核中斷進(jìn)入和退出

    陷阱trap 術(shù)語(yǔ)陷阱描述了軟件應(yīng)用程序中的控制轉(zhuǎn)移,其中陷阱處理通常在更特權(quán)的環(huán)境中執(zhí)行。例如,一個(gè)特定的 hart包含三種權(quán)限模式:機(jī)器、管理員和用戶(hù)。每種特權(quán)模式都有自己的軟件執(zhí)
    的頭像 發(fā)表于 10-08 09:47 ?484次閱讀
    U54內(nèi)核中斷進(jìn)入和退出

    Lombok的使用

    在平時(shí)我們工作的時(shí)候,我們經(jīng)常會(huì)使用 toString() 方法來(lái)輸出一個(gè)對(duì)象的一些屬性信息。Lombok 給我們提供了一個(gè)自動(dòng)生成 toString() 代碼的注解,可以減少代碼行數(shù),如果代碼屬性
    的頭像 發(fā)表于 09-25 14:03 ?633次閱讀

    Lombok版的構(gòu)造器注入方式

    做開(kāi)發(fā)的同學(xué)可能都會(huì)發(fā)現(xiàn), IDEA 在我們經(jīng)常使用的 @Autowired 注解上添加了警告: Field injection is not recommended, 即: 不推薦使用屬性注入。那怎么解決這個(gè)問(wèn)題呢? 問(wèn)題原因 違背單一職責(zé)原則 因?yàn)楝F(xiàn)在的業(yè)務(wù)一般都會(huì)使用很多依賴(lài), 但擁有太多的依賴(lài)通常意味著承擔(dān)更多的責(zé)任,而這顯然違背了單一職責(zé)原則。 依賴(lài)Spring @Autowired 由 Spring 提供,而 @Resource 是JSR-250提供的,它是Java標(biāo)準(zhǔn)。前者會(huì)警告,而后者不警告,就是因?yàn)榍罢邔?dǎo)致了應(yīng)用與框架的強(qiáng)綁定,若
    的頭像 發(fā)表于 09-25 10:50 ?818次閱讀