Groovy - 探索之 invokeMethod 方法

简介: Groovy - 探索之 invokeMethod 方法


我们知道,在Java语言中,所有的Java类都继承了Object对象。通过Object对象,所有的Java类都隐藏的实现了“equals”等方法。同样,在Groovy语言中,所有的Groovy类都隐藏的实现了GroovyObject接口,这样,我们的Groovy类就隐藏的实现了很多的方法,如“isCase”等。

这篇文字要谈谈的就是GroovyObject接口的“invokeMethod”方法,这个方法对于我们Groovy语言的动态性编程很有帮助,可以帮助我们实现一些很有时代性的功能,比如DSL。本文就是要谈谈“invokeMethod”方法的基础,通过这个基础,我们才可以通向DSL编程。

首先,我们来看看“invokeMethod”方法在一个Groovy类中的作用,先来看一个例子:

classInvokeMethodTestor {
deftest()
    {
println'hello,function name is test'    }
definvokeMethod(Stringname,Objectargs)
    {
println"the other function, name is ${name}"    }
}

这是一个很简单的Groovy类,我们有一个test方法,用来向控制台打印一句话;然后我们实现了invokeMethod方法,并且把参数“name”打印在控制台。

在讲述“invokeMethod”方法的作用之前,我们先来测试一下上面的类。

deftestor=newInvokeMethodTestor()
testor.test()
testor.hello()
testor.doSomething()

我们先来看看测试结果:

hello,function name is test

the other function, name is hello

the other function, name is doSomething

通过测试结果,我们可以看出,语句“testor.test()”调用了“InvokeMethodTestor”类的“test”方法,而语句“testor.hello()”和“testor.doSomething()”却都调用了“InvokeMethodTestor”类的“invokeMethod”方法。

这就告诉我们,对于一个实现了“invokeMethod”方法的Groovy类的对象,可以执行任意的方法,如果该方法已经在该类中实现,就调用该方法,如“testor.test()”就调用“InvokeMethodTestor”类的“test”方法;如果该方法没有在该类中实现,如“testor.hello()”和“testor.doSomething()”,就调用该类的“invokeMethod”方法。

这样说来,“invokeMethod”方法其实蛮简单的,一点都不神秘和麻烦。但是它的作用却一点都不能小觑,它给我们的Groovy语言编程带来了很大的动态性。

下面试着举一个小小的例子说明。

比如,我们有一个Student类,里面放的是学生的成绩,如“语文”、“数学”、“英语”等等,如下:

classStudent {  
Stringno;
Stringname;
floatchinScore;
floatmathScore;
floatenglScore;
floatphysScore;
floatchemScore;
floattotalScore;
}

同时,我们有一些学生(已经记录的各科成绩)在一个List对象里,如下:

Listscores= [newStudent(no:'123',name:'Tom',chinScore:90,mathScore:99,englScore:60,physScore:88,chemScore:96)]
scores<<newStudent(no:'124',name:'Mike',chinScore:88,mathScore:90,englScore:90,physScore:98,chemScore:87)
scores<<newStudent(no:'125',name:'Alice',chinScore:100,mathScore:55,englScore:98,physScore:67,chemScore:56)

这些学生在List对象里是以学号排序的,我们来看看:

1. scores.each{
2.     println it.name+' : '+it.no
3. }

结果为:

Tom : 123

Mike : 124

Alice : 125

可以看到的确如此。

下面,我们的语文老师希望以语文成绩排序,而数学老婆希望以数学成绩排序,英语老师则希望以英语成绩排序,……,班主任则希望以总分排序。

看到这里,你可以会说,我做一个方法来实现所有老师的愿望,这个方法有两个参数,一个是List对象,一个是type,type参数用来表示语文老师要的语文,数学老师要的数学等等。

这个方法当然是不错的,但不是最酷的。最酷的方法是语文老师调用sortByChinScore()方法,而数学老师调用sortByMathScore()方法,英语老师调用sortByEnglScore()方法,等等。

且慢!这不是要我写六七个方法来实现所有老师的要求?这未免有点无聊吧?像这样的代码写起来也枯燥乏味呀。

当然不用写六七个方法,答案就是“invokeMethod”方法。我们来看看是如何实现这样一个想法的:

importjava.util.Collectionsimportjava.util.ComparatorclassSortHelper{
deflistpublicSortHelper(list)
    {
this.list=list    }
// 所有的以sort开头的方法都来调用“invokeMethod”,当然,其他方法也有可能来调用它,但我不做处理。definvokeMethod(Stringname,Objectargs)
    {
//首先判断方法名是否以“sortBy”开头,是则处理,否则不处理。if(name.indexOf('sortBy')==0)
       {
//取得属性名,如“ChinScore”name=name[6..name.length()-1]
//把第一个字母由大写变小写,就取得了属性名name=name[0].toLowerCase()+name[1..name.length()-1]
//实现Comparator接口,大家可以参考jdk文档。defcomparator= {
node1,node2->returnnode1."${name}".compareTo(node2."${name}")
           } asComparator//排序Collections.sort(this.list,comparator) 
       }
    }
}

真的很简单。下面我们来测试一些这个类:

defsorter=newSortHelper(scores)
sorter.sortByChinScore()
scores.each{
printlnit.name}

结果为:

Mike

Tom

Alice

从语文成绩来看,Mike 88,Tom 90,Alice 100。排序是正确的。下面我们以数学成绩排序:

1. sorter.sortByMathScore()
2.  scores.each{
3.    println it.name
4. }

结果为:

Alice

Mike

Tom

从数学成绩来看,Alice 55,Mike 90,Tom 99,排序也没有问题。大家可以再测几个看看。

像这样的功能的确够酷,我们今后在Groovy/Grails平台会经常遇到;同时,这样一个思想也可以发展成DSL的一种实现。

目录
相关文章
|
Java 编译器
【Java用法】Java关键字 —— instanceof
【Java用法】Java关键字 —— instanceof
98 0
|
8月前
|
Java
什么是Java匿名类?
【4月更文挑战第16天】
202 9
什么是Java匿名类?
|
存储 安全 Java
深入了解Java中的StringJoiner类
`StringJoiner`是Java 8引入的一个用于拼接字符串的工具类。它允许我们以指定的分隔符将一组字符串连接成一个单独的字符串。我们可以在其源码中看到一个熟悉的身影`StringBuilder`,它是基于`StringBuilder`实现字符串操作的,所以它的性能和`StringBuilder`是差不多的,它也是非线程安全的。
192 3
深入了解Java中的StringJoiner类
|
Java
JAVA_equals方法重写
JAVA_equals方法重写
56 0
|
XML 敏捷开发 IDE
Groovy 的介绍和简单的使用
Groovy 的介绍和简单的使用
Java StringJoiner类
Java StringJoiner类学习
Java StringJoiner类
java175-method类反射机制
java175-method类反射机制
145 0
java175-method类反射机制
【错误记录】Groovy 函数拦截调用 invokeMethod 导致栈溢出 ( java.lang.StackOverflowError )
【错误记录】Groovy 函数拦截调用 invokeMethod 导致栈溢出 ( java.lang.StackOverflowError )
255 0
【错误记录】Groovy 函数拦截调用 invokeMethod 导致栈溢出 ( java.lang.StackOverflowError )