嘴上說(shuō)反對(duì)Lombok,身體卻很誠(chéng)實(shí)…
我見(jiàn)過(guò)很多反對(duì) Lombok 的同學(xué),背地里又偷偷的把插件添加了進(jìn)去,這是真香原理在搞鬼。
圖片來(lái)自 Pexels
嘴上說(shuō)不要,身體很誠(chéng)實(shí)。反對(duì)的人,應(yīng)該是沒(méi)見(jiàn)過(guò)一些業(yè)務(wù)代碼的冗長(zhǎng)繁雜,還沉浸在自己病態(tài)的完美主義中。要面對(duì)又臟又亂的從業(yè)環(huán)境,面對(duì)現(xiàn)實(shí)。
Lombok 可以消除 Java 的冗長(zhǎng),減少代碼的長(zhǎng)度,讓關(guān)注點(diǎn)轉(zhuǎn)移到該專注的地方。
SpringBoot 把 Lombok 放到了它的依賴中,Java14 甚至也借鑒了這種思想,推出了 record 語(yǔ)法。
就是類似于下面這種:
- record Point(int x, int y) { }
本篇文章,不打算討論什么類似于 @Data 注解之類的。我們討論一個(gè)比較偏門的,但是又讓你感覺(jué)相見(jiàn)恨晚的一個(gè)注解:RequiredArgsConstructor。
爆炸的屬性注入
Spring 提供了兩種注入模式,這也是非常初級(jí)的程序員經(jīng)常被問(wèn)到的三種 DI 寫法。
一種是屬性注入(Filed injection),一種是通過(guò) Setter 方法,一種是構(gòu)造器注入。
霍霍,我撒謊了,經(jīng)常被問(wèn)的是 byName 和 byType。不過(guò),這年頭,我們用的跟多的是 @Autowired 注解。
代碼寫起來(lái)一般是這樣的:
- @Service
- public class GoodsServiceImpl implements GoodsSrv {
- @Autowired
- private GoodsRepo goodsRepo;
- @Autowired
- private TagRepo tagRepo;
- @Autowired
- private TagRefRepo tagRefRepo;
- @Autowired
- private BrandRepo brandRepo;
- @Autowired
- private UnitRepo unitRepo;
- }
這一般沒(méi)什么問(wèn)題,因?yàn)樽⑷氲淖侄问怯邢薜摹5绻銢](méi)見(jiàn)過(guò)一些項(xiàng)目代碼,你會(huì)被這種程序界完美的表象給蒙騙了。
業(yè)務(wù)代碼,不加注釋,單文件長(zhǎng)度超過(guò) 2000 行的比比皆是。注入的屬性能達(dá)到十幾個(gè)之多。這部分注入代碼真是臟亂差。
不僅如此,這些字段,還會(huì)在 IDE 里變成灰色,告訴你未被初始化,代碼變成了丑八怪。
事實(shí)上,Spring 從 4.0 開(kāi)始, 就 不 推 薦 使 用 屬 性 注 入 模 式 了 ,原因是它可以讓我們忽略掉一些代碼可能變壞的隱患。你可以自行搜索這個(gè)問(wèn)題,我們也不展開(kāi)說(shuō)了。
既然 Spring 推薦使用顯示的 Setter 和構(gòu)造器方式,那我們就切換一下實(shí)現(xiàn)方案。
Setter 方法基本上用的人比較少,因?yàn)樗映舾娱L(zhǎng)。要是給每一個(gè)屬性寫一個(gè) set 方法,我估計(jì)你即使用代碼生成器也玩吐了。
構(gòu)造器注入
那么,構(gòu)造器的方法就成了我們的首選。
樣例代碼如下:
- public class GoodsServiceImpl implements GoodsSrv {
- private GoodsRepo goodsRepo;
- private TagRepo tagRepo;
- private TagRefRepo tagRefRepo;
- private BrandRepo brandRepo;
- private UnitRepo unitRepo;
- public GoodsServiceImpl(
- GoodsRepo goodsRepo,
- TagRepo tagRepo,
- TagRefRepo tagRefRepo,
- BrandRepo brandRepo,
- UnitRepo unitRepo) {
- this.goodsRepo = goodsRepo;
- this.tagRefRepo = tagRefRepo;
- this.tagRefRepo = tagRefRepo;
- this.brandRepo = brandRepo;
- this.unitRepo = unitRepo;
- this.tagRepo = tagRepo;
- }
- }
Spring 不需要加入其他注解,就可以使用構(gòu)造器完成注入。問(wèn)題是,我們依然要寫很多代碼。
這個(gè)時(shí)候,你可能想到了 Lombok 的 AllArgsConstructor 注解。但它是針對(duì)于全部的屬性的,如果類中有一些非 Bean 的屬性,Spring 就會(huì)暈菜。
這個(gè)時(shí)候,就可以使用 RequiredArgsConstructor 了。
代碼如下:
- @Service
- @RequiredArgsConstructor
- public class GoodsServiceImpl implements GoodsSrv {
- final GoodsRepo goodsRepo;
- final TagRepo tagRepo;
- final TagRefRepo tagRefRepo;
- final BrandRepo brandRepo;
- final UnitRepo unitRepo;
- }
我們把需要注入的屬性,修改成 final 類型的(或者使用 @NotNull 注解,不推薦),這些屬性將構(gòu)成默認(rèn)的構(gòu)造器。
Java 要求 final 類型的屬性必須要初始化,如果沒(méi)有構(gòu)造方法代碼就會(huì)變紅。
我們可以看到修改之后的 IDE,惱人的灰色提示也消失了。
這樣的代碼,是非常簡(jiǎn)潔的。
更高級(jí)一點(diǎn)
RequiredArgsConstructor 注解,你還可以像下面這樣寫。即使是把 @__ 換成 @_,或者換成 @___,也是能正常的運(yùn)行。
- @RequiredArgsConstructor(onConstructor = @__(@Autowired))
它的意思是,給使用 Lombok 生成的構(gòu)造器方法,加入一個(gè) @Autowired 注解。
這是徹頭徹尾的 Lombok 語(yǔ)法,不過(guò)現(xiàn)在的 Spring 已經(jīng)不需要加入這樣的注解就能運(yùn)行了。
看我下面的代碼,是能實(shí)際運(yùn)行的。爽不爽?
- @RequiredArgsConstructor(onConstructor =
- @______________________________________(
- @Autowired
- ))
真是要命的美啊!
通過(guò)這些方式,你寫的代碼行數(shù),可能會(huì)急劇下降。在以代碼行數(shù)論貢獻(xiàn)的公司,可能會(huì)助你獲得 3.25,不過(guò)這 3.25 拿的驕傲。
作者:小姐姐味道
簡(jiǎn)介:聚焦基礎(chǔ)架構(gòu)和 Linux。十年架構(gòu),日百億流量,與你探討高并發(fā)世界,給你不一樣的味道。
編輯:陶家龍
出處:轉(zhuǎn)載自公眾號(hào)小姐姐味道(ID:xjjdog)