在 Java 和 Groovy 中創(chuàng)建和初始化列表的不同
我非常喜歡 Groovy 編程語言。我喜歡它是因?yàn)槲蚁矚g Java,盡管 Java 有時(shí)候感覺很笨拙。正因?yàn)槲沂悄敲聪矚g Java,其他運(yùn)行在 JVM 上語言都不能吸引我。比方說 Kotlin、Scala 還有 Clojure 語言,它們感覺上就和 Java 不一樣,因?yàn)樗鼈儗τ谑裁词呛玫木幊陶Z言的理解不同。Groovy 和它們都不一樣,在我看來,Groovy 是一個(gè)完美的選項(xiàng),特別是對于一部分程序員來說,他們喜歡 Java,但是又需要一個(gè)更靈活、更緊湊,并且有時(shí)候更直接的語言。
列表List 這種數(shù)據(jù)結(jié)構(gòu)是一個(gè)很好的例子,它可以容納一個(gè)無序的列表,列表中的元素可以是數(shù)字、字符串或者對象,程序員可以用某種方式高效地遍歷這些元素,特別是對于編寫和維護(hù)腳本的人來說,“高效”的關(guān)鍵就是要有簡潔清晰的表達(dá),而不需要一大堆“儀式”,把代碼的意圖都變模糊了。
安裝 Java 和 Groovy
Groovy 是基于 Java 的,因此需要同時(shí)安裝一個(gè) Java 才行。你的 Linux 發(fā)行版的倉庫中可能有最近的比較好的 Java 版本?;蛘?,你也可以在根據(jù) ??這些指導(dǎo)?? 來安裝 Groovy。對于 Linux 用戶來說,SDKMan 是一個(gè)不錯(cuò)的代替選項(xiàng),你可以使用它來獲取多個(gè) Java 和 Groovy 版本,以及許多其他的相關(guān)工具。在這篇文章中,我使用的 SDK 發(fā)行版是:
- Java: OpenJDK 11 的 11.0.12-open 版本
- Groovy: 3.0.8 版本
言歸正傳
Java 中有很多方法可以實(shí)例化并初始化列表,從它最初被引入的時(shí)候就有了(我記得是在 Java 1.5 的時(shí)候,但請不要引用我的話)。在這些方法里,有兩個(gè)有趣的方法,它們涉及到了 ??java.util.Arrays?
? 和 ??java.util.List?
? 這兩個(gè)類。
使用 java.util.Arrays 類
??java.util.Arrays?
? 類定義了一個(gè) ??asList()?
? 靜態(tài)方法,它可以被用來創(chuàng)建一個(gè)基于數(shù)組的列表,因此大小是不可變的,盡管其中的元素是可以被修改的。下面是它的使用方式:
var a1 = Arrays.asList(1,2,3,4,5,6,7,8,9,10); // immutable list of mutable elements
System.out.println("a1 = " + a1);
System.out.println("a1 is an instance of " + a1.getClass());
// output is
// a1 = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
// a1 is an instance of class java.util.Arrays$ArrayList
a1.set(0,0); // succeeds
System.out.println("a1 = " + a1); // output is
// a1 = [0, 2, 3, 4, 5, 6, 7, 8, 9, 10]
a1.add(11); // fails producing
// Exception in thread "main" java.lang.UnsupportedOperationException
System.out.println("a1 = " + a1); // not reached
使用 java.util.List 類
??java.util.List?
? 類定義了一個(gè) ??of()?
? 靜態(tài)方法,它可以被用來創(chuàng)建一個(gè)不可變的列表,其中的元素是否可變要取決于它們本身是否支持修改。下面是它的使用方式:
var a2 = List.of(1,2,3,4,5,6,7,8,9,10);
System.out.println("a2 = " + a2);
System.out.println("a2 is an instance of " + a2.getClass());
// output is
// a2 = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
// a2 is an instance of class java.util.ImmutableCollections$ListN
a2.set(0,0); // fails producing
// Exception in thread "main" java.lang.UnsupportedOperationException
System.out.println("a2 = " + a2); // not reached
a2.add(11); // also fails for same reason if above two lines commented out
System.out.println("a2 = " + a2); // not reached
因此,我可以使用 ??Arrays.asList()?
?,也可以使用 ??List.of()?
? 方法,前提是如果我想要的是一個(gè)大小不能改變、且不關(guān)心元素是否可變的列表。
如果我想要初始化一個(gè)可變的列表,我更傾向于把這些不可變的列表作為參數(shù)傳給一個(gè)列表構(gòu)造器,就像下面這樣:
var a1 = new ArrayList<Integer>(Arrays.asList(1,2,3,4,5,6,7,8,9,10));
System.out.println("a1 = " + a1);
System.out.println("a1 is an instance of " + a1.getClass());
// output is
// a1 = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
// a1 is an instance of class java.util.ArrayList
a1.set(0,0);
System.out.println("a1 = " + a1);
//output is
// a1 = [0, 2, 3, 4, 5, 6, 7, 8, 9, 10]
a1.add(11);
System.out.println("a1 = " + a1);
// output is
// a1 = [0, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
注意,這個(gè) ??Arrays.asList()?
? 方法是用來初始化這個(gè)新的 ??ArrayList<Integer>()?
? 的,也就是說,它為這個(gè)傳進(jìn)來的列表創(chuàng)建了一個(gè)可變的拷貝。
現(xiàn)在,或許只有我這么想,但是這種方式確實(shí)看起來需要理解很多關(guān)于 ??java.util.Arrays?
? 和 ??java.util.List?
? 類的細(xì)節(jié)才行,而我只是想要?jiǎng)?chuàng)建并初始化一個(gè)數(shù)字列表而已(盡管真正使用到的語句并沒有太多“儀式”)。下面是真正用到的那行代碼,僅供參考:
var a1 = new ArrayList<Integer>(Arrays.asList(1,2,3,4,5,6,7,8,9,10));
Groovy 是怎么做的
下面來看看在 Groovy 中如何實(shí)現(xiàn)上述需求:
def a1 = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
println "a1 = $a1"
println "a1 is an instance of ${a1.getClass()}"
// output is
// a1 = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
// a1 is an instance of class java.util.ArrayList
a1[0] = 0
println "a1 = $a1"
// output is
// a1 = [0, 2, 3, 4, 5, 6, 7, 8, 9, 10]
a1 << 11
println "a1 = $a1"
// output is
// a1 = [0, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]
我們一眼就能發(fā)現(xiàn),Groovy 使用了 ??def?
? 關(guān)鍵字而不是 ??var?
? 關(guān)鍵字。我還發(fā)現(xiàn)了,僅僅是把一系列的類型(在這個(gè)例子里是整數(shù))放進(jìn)括號里,我就得到了一個(gè)創(chuàng)建好的列表。此外,這樣創(chuàng)建出來的列表完全就是我想要的:一個(gè)可變的 ??ArrayList?
? 實(shí)例。
現(xiàn)在,或許再一次只有我這么想,但是上面的代碼看起來要簡單多得多 —— 不用記住 ??.of()?
? 和 ??.asList()?
? 返回的是“半不變semi-mutable”的結(jié)果,也不用為它們做一些補(bǔ)償。另外一個(gè)好處是,我現(xiàn)在可以使用括號和下標(biāo)來引用列表中的某個(gè)特定元素,而不用這個(gè)叫 ??set()?
? 方法。另外,這個(gè)跟在列表后面的 ??<<?
? 操作符也很方便,我再也不用調(diào)用 ??add()?
? 方法來添加元素啦。還有,你注意到代碼中沒有分號了嗎?沒錯(cuò),在 Groovy 里,句末的分號并不是必須的。最后,我們來看看字符串插值,只要在字符串里用 ??$變量?
? 或者 ??${表達(dá)式}?
? 就可以實(shí)現(xiàn)了哦!
在 Groovy 世界中還藏著許多“有待發(fā)掘”的東西。上面的列表定義其實(shí)是一個(gè)動(dòng)態(tài)類型(Groovy 中默認(rèn))和 Java 中的靜態(tài)類型的對比。在上面的 Groovy 代碼定義的那一行,變量 ??a1?
? 的類型是在運(yùn)行的時(shí)候,根據(jù)等號右邊的表達(dá)式的計(jì)算結(jié)果推斷出來的?,F(xiàn)在我們都知道,動(dòng)態(tài)語言可以給我們帶來強(qiáng)大的功能,有了強(qiáng)大的功能,我們有了很多機(jī)會(huì)去嘗試不同的東西。對于那些不喜歡動(dòng)態(tài)類型的程序員來說,Groovy 也支持靜態(tài)類型。
Groovy 相關(guān)資源
Apache Groovy 網(wǎng)站上有非常多的文檔。另一個(gè)很棒的 Groovy 資源是 ??Mr. Haki??。學(xué)習(xí) Groovy 還有一個(gè)很棒的原因,那就是可以接著學(xué)習(xí) ??Grails??,后者是一個(gè)優(yōu)秀的、高效率的全棧 Web 框架,基于許多優(yōu)秀組件構(gòu)建而成,比如有 Hibernate、Spring Boot 和 Micronaut 等。