Groovy学习笔记(三):接口,布尔判断,操作符重载
Groovy接口Groovy 不需要显示的通过new创建匿名内部类的实例。 //Button对象
class Button {
void addOnClickListener(OnClickListener listener) {
listener.onClick()
}
void addOnLongClickListener(OnLongClickListener listener) {
listener.onLongClick()
}
}
//按钮的点击监听
interface OnClickListener {
void onClick()
}
//长按事件监听
interface OnLongClickListener {
void onLongClick()
}
调用了addOnClickListener方法,同时为该方法提供了一个代码块,借助as操作符,相当于实现了OnClickListener接口: def button = new Button()
listener = { println 'addListener' }
button.addOnClickListener(listener) as OnClickListener
输出:
Groovy自会处理剩下的工作。它会拦截对接口中任何方法的调用,然后将调用路由到我们提供的代码块。 对于有多个方法的接口,如果打算为其所有方法提供一个相同的实现,和上面一样,不需要特殊的操作: def button = new Button()
listener = { println 'addListener' }
button.addOnClickListener(listener) as OnClickListener
button.addOnLongClickListener(listener) as OnLongClickListener
输出:
Groovy没有强制实现接口中的所有方法:可以只定义自己关心的,而不考虑其他方法。如果剩下的方法从来不会被调用,那也就没必要去实现这些方法了。当在单元测试中通过实现接口来模拟某些行为时,这项技术非常有用。 但是在大多数实际情况下,接口中的每个方法需要不同的实现。不用担心,Groovy可以摆平。只需要创建一个映射,以每个方法的名字作为键,以方法对应的代码体作为键值,同时使用简单的Groovy风格,用冒号(:)分隔方法名和代码块即可。 不必实现所有方法,只需实现真正关心的那些即可。如果未予实现的方法从未被调用过,那么也就没有必要浪费精力去实现这些伪存根。当然,如果没提供的方法被调用了,则会出现异常: class Button { void addOnStateChangeListener(OnStateChangeListener listener) { listener.onCreate() listener.onStart() listener.onStop() listener.onDestory() } } //状态监听 interface OnStateChangeListener { void onCreate() void onStart() void onStop() void onDestroy() }
def onStateChangelistener = [
onCreate: {
println 'onCreate'
},onStart : {
println 'onStart'
},onStop : {
println 'onStop'
} //这里只实现3个状态监听,onDestroy()并未实现
]
button.addOnStateChangeListener(onStateChangelistener as OnStateChangeListener)
结果: 布尔值判断Java要求if语句的条件部分必须是一个布尔表达式,比如前面例子中的if(obj!=null)和if(val>0)。 Groovy会尝试推断,如果在需要布尔值的地方放了一个对象引用,Groovy会检查该引用是否为null。它将null视作false,将非null的值视作true: str = 'hello'
if (str) {
println str
}
结果: 更准确的说,表达式的结果还与对象的类型有关。例如,如果对象是一个集合(如java.util.ArrayList),那么Groovy会检查该集合是否为空。 str1 = null
println str1 ? 'str1 is not null' : 'str1 is null'
str2 = [1,2,3]
println str2 ? 'str2 is not null' : 'str2 is null'
str3 = []
println str3 ? 'str3 is not null' : 'str3 is null'
结果: 集合类不是唯一受到特殊对待的。那么有哪些类型将被特殊对待,Groovy又是如何计算它们的呢?
除了使用Groovy内建的布尔求值约定,在自己的类中,还可以通过实现asBoolean()方法来编写自己的布尔转换。 操作符重载Groovy支持操作符重载,可以巧妙地应用这一点来创建DSL 比如我们可以重载++操作符,该示例映射的是String类的next()方法: for (ch = 'a'; ch < 'd'; ch++) { println ch }
结果:
Groovy中还可以使用简洁的for-each语法,不过两种实现都用到了String类的next()方法: for (ch in 'a'..'c') {
println ch
}
结果同上,都是输出了abc。 要向集合中添加元素,可以使用<<操作符,该操作符会被转换为Groovy在Collection上添加的leftShift()方法: list = ['hello']
list << 'groovy!'
println list
输出:
通过添加映射方法,我们可以为自己的类提供操作符,比如为+操作符添加plus()方法: class B03ComplexNumber {
def real,imaginary //实部,虚部
def plus(other) {
new B03ComplexNumber(real: real + other.real,imaginary: imaginary + other.imaginary)
}
@Override
String toString() {
return "$real ${imaginary > 0 ? '+' : ''} ${imaginary}i "
}
}
我们执行这段代码: c1 = new B03ComplexNumber(real: 1,imaginary: 4)
c2 = new B03ComplexNumber(real: 2,imaginary: 3)
println c1 + c2
输出:
ComplexNumber类重载了+操作符。对于计算涉及负数平方根的复杂方程式,复数非常有用: 复数有实部和虚部。 因为在ComplexNumber类上添加了plus()方法,所以可以使用+操作符把两个复数加到一起,得到又一个作为结果的复数。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |