Groovy学习笔记3:接口,布尔判断,操作符重载

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

输出:

addListener

Groovy自会处理剩下的工作。它会拦截对接口中任何方法的调用,然后将调用路由到我们提供的代码块。

对于有多个方法的接口,如果打算为其所有方法提供一个相同的实现,和上面一样,不需要特殊的操作:

def button = new Button()

listener = { println 'addListener' }

button.addOnClickListener(listener) as OnClickListener
button.addOnLongClickListener(listener) as OnLongClickListener

输出:

addListener
addListener

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)

结果:

Groovy学习笔记3:接口,布尔判断,操作符重载_第1张图片
image.png

布尔值判断

Java要求if语句的条件部分必须是一个布尔表达式,比如前面例子中的if(obj!=null)和if(val>0)。

Groovy会尝试推断,如果在需要布尔值的地方放了一个对象引用,Groovy会检查该引用是否为null。它将null视作false,将非null的值视作true:

str = 'hello'

if (str) {
    println str
}

结果:

Groovy学习笔记3:接口,布尔判断,操作符重载_第2张图片
image.png

更准确的说,表达式的结果还与对象的类型有关。例如,如果对象是一个集合(如java.util.ArrayList),那么Groovy会检查该集合是否为空。
因此,在这种情况下,只有当obj不为null,而且该集合至少包含一个元素时,表达式if(obj)才会被计算为true:

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学习笔记3:接口,布尔判断,操作符重载_第3张图片
image.png

集合类不是唯一受到特殊对待的。那么有哪些类型将被特殊对待,Groovy又是如何计算它们的呢?

类型 为真的条件
Boolean true
Collection 集合不为空
character 值不为0
CharSequence 长度不为0
Enumration hasMoreElements()为true
Iterator hasNext()为true
Number Double值部位0
Map 映射不为空
Matcher 至少有一个匹配
Object[] 长度大于0
其他引用类型 引用不为null

除了使用Groovy内建的布尔求值约定,在自己的类中,还可以通过实现asBoolean()方法来编写自己的布尔转换。

操作符重载

Groovy支持操作符重载,可以巧妙地应用这一点来创建DSL

比如我们可以重载++操作符,该示例映射的是String类的next()方法:

for (ch = 'a'; ch < 'd'; ch++) {
    println ch
}

结果:

a
b
c

Groovy中还可以使用简洁的for-each语法,不过两种实现都用到了String类的next()方法:

for (ch in 'a'..'c') {
    println ch
}

结果同上,都是输出了abc。

要向集合中添加元素,可以使用<<操作符,该操作符会被转换为Groovy在Collection上添加的leftShift()方法:

list = ['hello']
list << 'groovy!'
println list

输出:

[hello, groovy!]

通过添加映射方法,我们可以为自己的类提供操作符,比如为+操作符添加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

输出:

3 + 7i

ComplexNumber类重载了+操作符。对于计算涉及负数平方根的复杂方程式,复数非常有用: 复数有实部和虚部。
因为在ComplexNumber类上添加了plus()方法,所以可以使用+操作符把两个复数加到一起,得到又一个作为结果的复数。

你可能感兴趣的:(Groovy学习笔记3:接口,布尔判断,操作符重载)