Groovy 是基于 JVM 虛擬機(jī)的一種動態(tài)語言沐兰,它的語法和 Java 非常相似趋翻,由 Java 入門Groovy芹壕,基本上沒有任何障礙窟绷。Groovy 完全兼容 Java锯玛,又在此基礎(chǔ)上增加了很多動態(tài)類型和靈活的特性,比如支持閉包兼蜈,支持 DSL攘残,可以說它是一門非常靈活的動態(tài)腳本語言。
Groovy 的特性雖然不多为狸,但也有一些歼郭,我們不可能在這里都講完,這也不是這本書的初衷辐棒,在這里我挑一些和Gradle有關(guān)的知識講病曾,讓大家很快的入門 Groovy,并且能看懂這門腳本語言漾根,知道在Gradle為什么這么寫知态。其次是每個 Gradle 的 build 腳本文件都是一個 Groovy 腳本文件,你可以在里面寫任何符合 Groovy 的代碼立叛,比如定義類,生命函數(shù)贡茅,定義變量等等秘蛇,而 Groovy 又完全兼容Java,這就意味著你可以在build腳本文件里寫任何的Java代碼顶考,非常靈活方便赁还。
字符串
字符串,每一門語言都會有對字符串的處理驹沿,Java相對要稍微復(fù)雜一些艘策,限制比較多,相比而言渊季,Groovy 非常方便朋蔫,比如字符串的運(yùn)算罚渐、求值、正則等等驯妄。
從現(xiàn)在開始我們算是正式的介紹 Groovy 了荷并,在此之前我們先要知道,在 Groovy 中青扔,分號不是必須的源织。相信很多用Java的朋友都習(xí)慣了,每一行的結(jié)束必須有分號微猖,但是 Groovy 沒這個強(qiáng)制規(guī)定谈息,所以你看到的Gradle腳本很多都沒有分號,其實(shí)這個是 Groovy 的特性凛剥,而不是 Gradle 的侠仇。沒有分號的時候,我們閱讀的時候每一行默認(rèn)為有分號就好了当悔。
在Groovy中傅瞻,單引號和雙引號都可以定義一個字符串常量(Java里單引號定義一個字符),不同的是單引號標(biāo)記的是純粹的字符串常量盲憎,而不是對字符串里的表達(dá)式做運(yùn)算嗅骄,但是雙引號可以。
- 單引號沒有運(yùn)算的能力饼疙,它里面的所有都是常量字符串溺森。
- 雙引號可以直接進(jìn)行表達(dá)式計(jì)算的這個能力非常好用,我們可以用這種方式進(jìn)行字符串鏈接運(yùn)算窑眯,再也不用 Java 中繁瑣的 +號了屏积。記住這個嵌套的規(guī)則,一個美元符號緊跟著一對花括號磅甩,花括號里放表達(dá)式炊林,比如
${name}
,${1+1}
等等,只有一個變量的時候可以省略花括號卷要,比如$name
渣聚。
task printStringVar << {
def str1 = "我是火車王"
println"$str1"
println"${str1}, 誰敢召喚我, 你想借個${str1}"
}
集合
集合,也是我們在Java中經(jīng)常用到的僧叉,Groovy 完全兼容了Java的集合奕枝,并且進(jìn)行了擴(kuò)展,使得生命一個集合瓶堕,迭代一個集合隘道、查找集合的元素等等操作變得非常容易。常見的集合有 List、Set谭梗、Map 和 Queue忘晤,這里我們只介紹常用的 List 和 Map。
List
task list << {
def list = [1, 3, 5, 7, 9]
println list.getClass().name
println list[0]
println list[-1]//訪問最后一個元素
println list[-2]//訪問倒數(shù)第二個元素
println list[1..3]//訪問第2到第4個元素
// it 變量就是正在迭代的元素默辨,這里有閉包的知識
list.each {
println it
}
}
Map
task map << {
def map = ['width': 1366, 'height': 768]
println map.getClass().name
// 以下下方式都能快速的取出指定key的值
println map.width
println map["height"]
map.each{
println "${it.key}: ${it.value}"
}
}
對于集合德频,Groovy 還提供了諸如 collect、find缩幸、findAll 等便捷的方法壹置,有興趣的朋友可以找相關(guān)文檔。
方法
- 括號是可以省略的
- return是可以不寫的
task testMethod <<{
def i1 = 12
def i2 = 67
//括號, 分號都不要了
printSum i1, i2
def maxResult = getMax i1, i2
println maxResult
}
// 無 return
def printSum(int i1, int i2){
println i1+i2
}
// 有 return 值, 為最后一句為返回值
def getMax(int i1, int i2) {
def max = i1;
if (i2 > i1) {
max = i2;
}
"max is $max"
}
代碼塊是可以作為參數(shù)傳遞的
代碼塊--一段被花括號包圍的代碼表谊,其實(shí)就是我們后面要將的閉包钞护,Groovy是允許其作為參數(shù)傳遞的,但是結(jié)合這我們上面方法的特性爆办,最后的基于閉包的方法調(diào)用就會非常優(yōu)雅难咕、易讀。以我們的集合的each方法為例距辆,它接受的參數(shù)其實(shí)就是一個閉包余佃。
JavaBean
task testJavaBean << {
Person p = new Person();
p.name = "砰砰博士"
println p.name
println "${p.name}"
println "${p.age}"
println "${p.brand}" //能這么用, 其實(shí)只是因?yàn)樵搶ο罄锒x了相應(yīng)的getter/setter方法而已
}
class Person {
String name;
private int age;
public String getBrand(){
'hearthstone'
}
}
閉包
閉包是Groovy的一個非常重要的特性,可以說他是DSL的基礎(chǔ)跨算。閉包不是Groovy的首創(chuàng)爆土,但是它支持這一重要特性,這就使用我們的代碼靈活诸蚕、輕量步势、可復(fù)用,再也不用像Java一樣動不動就要搞一個類了背犯,雖然Java后來有了匿名內(nèi)部類坏瘩,但是一樣冗余不靈活。
初識閉包
前面我們講過漠魏,閉包其實(shí)就是一段代碼塊倔矾,下面我們就一步步實(shí)現(xiàn)自己的閉包,了解閉包的it變量的由來柱锹。集合的 each 方法我們已經(jīng)非常熟悉了破讨,我們就以其為例,實(shí)現(xiàn)一個類似的閉包功能奕纫。
在上面的例子中我們定義了一個方法customEach,它只有一個參數(shù)烫沙,用于接收一個閉包(代碼塊)匹层,那么這個閉包如何執(zhí)行呢?很簡單,跟一對括號就是執(zhí)行了升筏,會JavaScript的朋友是不是覺得很熟悉撑柔,把它當(dāng)做一個方法調(diào)用,括號里的參數(shù)就是該閉包接收的參數(shù)您访,如果只有一個參數(shù)铅忿,那么就是我們的it變量了。
向閉包傳遞參數(shù)
當(dāng)閉包有一個參數(shù)時灵汪,默認(rèn)就是it檀训;當(dāng)有多個參數(shù)是,it就不能表示了享言,我們需要把參數(shù)一一列出峻凫。
task helloClosure << {
customEachMap{k,v ->
println "${k}: ${v}"
}
}
def customEachMap(closure){
def map1 = ['張三': 18, '李四': 20, '老五': 25]
map1.each{
closure it.key, it.value
}
}
閉包委托
Groovy閉包的強(qiáng)大之處在于它支持閉包方法的委托。Groovy的閉包有thisObject览露、owner荧琼、delegate三個屬性,當(dāng)你在閉包內(nèi)調(diào)用方法時差牛,由他們來確定使用哪個對象來處理命锄。默認(rèn)情況下delegate和owner是相等的,但是delegate是可以被修改的偏化,這個功能是非常強(qiáng)大的脐恩,Gradle中的很閉包的很多功能都是通過修改delegate實(shí)現(xiàn)的。
task testDelegate << {
new Delegate().test{
println thisObject.getClass().name
println owner.getClass().name
println delegate.getClass().name
method1()
it.method1()
}
}
def method1(){
println "Context this: ${this.getClass().name} in root, method1 in root"
}
class Delegate{
def method1(){
println "Context this: ${this.getClass().name} in Delegate, method1 in Delegate"
}
def test(Closure<Delegate> closure){
closure(this)
}
}
運(yùn)行我們可以看到輸出:
閉包內(nèi)方法的處理順序是thisObject>owner>delegate夹孔。
DSL
DSL(Domain Specific Language),領(lǐng)域特定語言被盈,說白了就是專門關(guān)注某一領(lǐng)域?qū)iT語言,在于專搭伤,而不是全只怎,所以才叫領(lǐng)域特定的,而不是像Java這種通用全面的語言怜俐。
Gradle 就是一門 DSL身堡,他是基于 Groovy 的,專門解決自動化構(gòu)建的DSL拍鲤。自動化構(gòu)建太復(fù)雜贴谎、太麻煩、太專業(yè)季稳,我們理解不了擅这,沒問題,專家們就開發(fā)了DSL--Gradle景鼠,我們作為開發(fā)者只要按照 Gradle DSL 定義的仲翎,書寫相應(yīng)的 Gradle 腳本就可以達(dá)到我們自動化構(gòu)建的目的,這也是DSL的初衷。
DSL 涉及的東西還有很多溯香,這里我們簡單的提一下概念鲫构,讓大家有個了解,關(guān)于這方便更詳細(xì)的可以閱讀世界級軟件開發(fā)大師Martin Fowler的《領(lǐng)域特定語言》玫坛,這本書介紹的非常詳細(xì)结笨。
參考
本文純屬自學(xué)歷程 + 一些記錄,絕大部分內(nèi)容來自原書 Android Gradle權(quán)威指南湿镀。覺得對你有用炕吸,請支持原書。