Gradle构建之Groovy语法详解

前言

前面几篇基本上对Gradle构建在Android中的使用场景和流程都有了大致了解,包括如何使用以及如何自定义插件等,这些基本都是基于官方的教程引导来了解的。接下来我们通过几篇文章来深入了解一下具体的几个核心知识点,今天就以Groovy语法为切入点,接下来几篇你可能还会看到关于标准的Gradle插件、Maven插件、自定义构建等具体的知识点相关文章。

Groovy初识

在Apache的groovy-lang.org官网可以看到一句很醒目的话:

A multi-faceted language for the Java platform / 一个Java平台的多窗口语言

从这一句话中我们可以得知它的重要性,它可以帮助我们方便的使Java平台和其他语言交互,它也是一个面向对象的语言。这门动态语言拥有类似Python、Ruby和Smalltalk中的一些特性,可以作为Java平台的脚本语言使用。由于其运行在JVM上的特性,Groovy可以使用其他Java语言编写的库。Groovy的语法与Java非常相似,大多数Java代码也符合Groovy的语法规则,尽管可能语义不同。

和Java的差异

Groovy语言在设计之初就让其可以很快速的让Java开发者使用,很多地方只会比Java更方便,而且让熟悉Java的你不感到别扭。

省略

Groovy语法允许省略分号和修改符。而且Groovy允许定义简单脚本,同时无需定义正规的class对象。

默认导包

默认情况下导入所有这些包和类,即您不必使用显式import语句来使用它们,所以在使用Groovy的时候不要导包。

1
2
3
4
5
6
7
8
java.io.*
java.lang.*
java.math.BigDecimal
java.math.BigInteger
java.net.*
java.util.*
groovy.lang.*
groovy.util.*

没有类型的Java代码

在Java中,如果要声明一个String变量,则必须输入:

1
String value = "Hello World";

但是,如果仔细想想,就会看出,等号右侧的字符已经表明value的类型是String。所以,Groovy允许省略value前面的String类型变量,并用def代替。

1
def value = "Hello World"

调用重载方法

在Groovy中在运行期间根据实际的参数类型来调用重载方法,而Java中在编译期间根据声明类型来调用重载方法。

1
2
3
4
5
6
7
8
9
int method(String arg) {  //声明为String类型参数
return 1;
}
int method(Object arg) { //声明为Object类型参数
return 2;
}

Object o = "Object"; //声明类型为Object,而实际类型是String
int result = method(o);

所以上面代码在Java中执行result的结果是2,而在Groovy中执行result的结果是1

默认public

我们知道在Java中省略类、方法、属性的可见修饰符默认为default,只有在同一个包内可见,而且不可继承。

作用域当前类同一个包子类其他包
public
protected
default
private

但是在Groovy中默认都是public修饰符,如果你想让字段只在同包内访问,可以使用@PackageScope修饰符。

1
2
3
class Person {
@PackageScope String name
}

内部类

静态内部类和匿名内部类与Java完全一致:

1
2
3
4
5
class A {
static class B {}
}

new A.B()

非静态内部类的唯一不同就是创建实例的不同,Groovy不支持y.new X()语法。相反,你必须写new X(y):

1
2
3
4
5
6
7
8
9
10
public class Y {
public class X {}
public X foo() {
return new X()
}
public static X createX(Y y) {
//return y.new X(); Java中可以这样创建内部类
return new X(y)
}
}

Lambda表达式

Groovy不支持Lambda表达式语法,但有闭包替代。

不一样的String

在Groovy中用单引号的字符串是和Java相同的String对象,而用双引号的字符串不一定是String对象,因为Groovy支持GString(也就是字符串中携带变量)

1
2
3
4
def arg = "123"
def iamstring = 'i am String'
def ianstr = "i am String"
def iamgstr = "i am GString $arg" //GString

不一样的双等于

在Java中使用==来比较两个基本类型或者对象的equals(),而在Groovy中==会被翻译成a.compareTo(b)==0,需要实现Comparable和equals方法。如果要检查是否是相同类型则可以使用a.is(b)

酷比了的循环

在Groovy中可以使用循环范围,是不是觉得很酷呢?

1
2
3
4
5
def repeat(val){
for(i in 1..5){ //从1开始循环到5(包含5),循环5次
println val
}
}

如果我们要上面的循环不包括5可以使用小于号,例如:for(i in 1..<5){ }

函数默认参数

Groovy的函数中可以给函数参数定义默认值,如下下面的repeat参数:

1
2
3
4
5
def repeat(val, repeat=5){
for(i in 0..<repeat){
println val
}
}

另外值得注意的是在Groovy中什么函数参数不需要类型或者def声明。

数组初始化

在Groovy中,{ …​ }块被保留用于闭包。这意味着您无法使用以下语法创建数组:

1
int[] array = { 1, 2, 3}

替代方式,也就是Groovy中的方法如下:

1
2
def array = [1,2,3]
def hash = [name:"Andy", age:45]

额外的关键字

Groovy中的关键字比Java中的更多。不要将它们用于变量名称。

1
2
3
4
as
def
in
trait

闭包

和Java中的Lambda表达式(可以参考《Lambda编程》)不同的是,在Groovy中使用的是闭包(Closure),它是Groovy中非常重要的一个数据类型或者说一种概念了。

闭包,是一种数据类型,它代表了一段可执行的代码。语法格式def xxx = {paramters -> code}举例如下:

1
2
3
4
5
def aClosure = {//闭包是一段代码,所以需要用花括号括起来..
param1, param2 -> //这个箭头很关键。箭头前面是参数定义,箭头后面是代码
println"this is code" //这是代码,最后一句是返回值,
//也可以使用return,和Groovy中普通函数一样
}

从C/C++语言的角度看,闭包和函数指针很像。闭包在Groovy中大量使用,比如很多类都定义了一些函数,这些函数最后一个参数都是一个闭包。

1
2
3
4
5
def acoll = ["Groovy", "Java", "Ruby"]

acoll.each{
println it
}

这段代码是不是有些奇怪呢?事实上这个each()是个函数,只不过省略了括号而已(Groovy中,当函数的最后一个参数是闭包的话,可以省略圆括号)。

这个时候机智的你是不是想到了android构建中的android{ },这个时候理解了它原来是一个android( { } )函数。

Groovy API

我们都知道Java语言有一套JDK来帮助我们快速开发,而Groovy语言也一样,API文档地址

为了方便Groovy语言更好的使用Java API,另外Groovy对JDK进行了包装,具体API文档请参考这里,我们可以利用Groovy的特性方便的使用Java API,这样极大的减少了我们的学习成本。

例如上面的each()方法就在java.util.List接口中定义了。

Groovy的each方法

可以看到这个Closure就是一个闭包对象基类,所以这里接收了一个闭包作为参数。

作为一门语言,Groovy是复杂的,是需要深入学习和钻研的。一本厚书甚至都无法描述Groovy的方方面面,庆幸的是在Gradle中我们使用的只是Groovy中的一些简单知识,上面这些知识基本够用了。

评论

Ajax Android AndroidStudio Animation Anroid Studio AppBarLayout Babel Banner Buffer Bulma ByteBuffer C++ C11 C89 C99 CDN CMYK COM1 COM2 CSS Camera Raw, 直方图 Chrome ContentProvider CoordinatorLayout C语言 DML DOM Dagger Dagger2 Darktable Demo Document DownloadManage ES2015 ESLint Element Error Exception Extensions File FileProvider Flow Fresco GCC Git GitHub GitLab Gradle Groovy HTML5 Handler HandlerThread Hexo Hybrid I/O IDEA IO ImageMagick IntelliJ Intellij Interpolator JCenter JNI JS Java JavaScript JsBridge Kotlin Lab Lambda Lifecycle Lint Linux Looper MQTT MVC MVP Maven MessageQueue Modbus Momentum MySQL NDK NIO NexT Next Nodejs ObjectAnimator Oracle VM Permission PhotoShop Physics Python RGB RS-232 RTU Remote-SSH Retrofit Runnable RxAndroid RxJava SE0 SSH Spring SpringBoot Statubar Task Theme Thread Tkinter UI UIKit UML VM virtualBox VS Code VUE ValueAnimator ViewPropertyAnimator Vue Web Web前端 Workbench api apk bookmark by关键字 compileOnly css c语言 databases demo hexo hotfix html iOS icarus implementation init jQuery javascript launchModel logo merge mvp offset photos pug query rxjava2 scss servlet shell svg tkinter tomcat transition unicode utf-8 vector virtual box vscode 七牛 下载 中介者模式 串口 临潼石榴 主题 书签 事件 享元模式 仓库 代理模式 位运算 依赖注入 修改,tables 光和色 内存 内核 内部分享 函数 函数式编程 分支 分析 创建 删除 动画 单例模式 压缩图片 发布 可空性 合并 同向性 后期 启动模式 命令 命令模式 响应式 响应式编程 图层 图床 图片压缩 图片处理 图片轮播 地球 域名 基础 增加 备忘录模式 外观模式 多线程 大爆炸 天气APP 太白山 头文件 奇点 字符串 字符集 存储引擎 宇宙 宏定义 实践 属性 属性动画 岐山擀面皮 岐山肉臊子 岐山香醋 工具 工厂模式 年终总结 开发技巧 异常 弱引用 恒星 打包 技巧 指针 插件 摄影 操作系统 攻略 故事 数据库 数据类型 数组 文件 新功能 旅行 旋转木马 时序图 时空 时间简史 曲线 杂谈 权限 枚举 架构 查询 标准库 标签选择器 样式 核心 框架 案例 桥接模式 检测工具 模块化 模板引擎 模板方法模式 油泼辣子 泛型 洛川苹果 浅色状态栏 源码 源码分析 瀑布流 热修复 版本 版本控制 状态栏 状态模式 生活 留言板 相册 相对论 眉县猕猴桃 知识点 码云 磁盘 科学 笔记 策略模式 类图 系统,发行版, GNU 索引 组件 组合模式 结构 结构体 编码 网易云信 网格布局 网站广播 网站通知 网络 美化 联合 膨胀的宇宙 自定义 自定义View 自定义插件 蒙版 虚拟 虚拟机 补码 补齐 表单 表达式 装饰模式 西安 观察者模式 规范 视图 视频 解耦器模式 设计 设计原则 设计模式 访问者模式 语法 责任链模式 贪吃蛇 转换 软件工程 软引用 运算符 迭代子模式 适配器模式 选择器 通信 通道 配置 链表 锐化 错误 键盘 闭包 降噪 陕西地方特产 面向对象 项目优化 项目构建 黑洞
Your browser is out-of-date!

Update your browser to view this website correctly. Update my browser now

×