本文共 3388 字,大约阅读时间需要 11 分钟。
我们知道,在Java语言中,所有的Java类都继承了Object对象。通过Object对象,所有的Java类都隐藏的实现了“equals”等方法。同样,在Groovy语言中,所有的Groovy类都隐藏的实现了GroovyObject接口,这样,我们的Groovy类就隐藏的实现了很多的方法,如“isCase”等。
这篇文字要谈谈的就是GroovyObject接口的“invokeMethod”方法,这个方法对于我们Groovy语言的动态性编程很有帮助,可以帮助我们实现一些很有时代性的功能,比如DSL。本文就是要谈谈“invokeMethod”方法的基础,通过这个基础,我们才可以通向DSL编程。
首先,我们来看看“invokeMethod”方法在一个Groovy类中的作用,先来看一个例子:
class InvokeMethodTestor { deftest() { println'hello,function name is test' } def invokeMethod(String name,Object args) { println"the other function, name is ${name}" } }
这是一个很简单的Groovy类,我们有一个test方法,用来向控制台打印一句话;然后我们实现了invokeMethod方法,并且把参数“name”打印在控制台。
在讲述“invokeMethod”方法的作用之前,我们先来测试一下上面的类。
def testor = new InvokeMethodTestor()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类,里面放的是学生的成绩,如“语文”、“数学”、“英语”等等,如下:
class Student { String no; String name; float chinScore; float mathScore; float englScore; float physScore; float chemScore; float totalScore;}
同时,我们有一些学生(已经记录的各科成绩)在一个List对象里,如下:
List scores = [new Student(no:'123',name:'Tom',chinScore:90,mathScore:99,englScore:60,physScore:88,chemScore:96)]scores<
这些学生在List对象里是以学号排序的,我们来看看:
scores.each{ println it.name+' : '+it.no}
结果为:
Tom : 123
Mike : 124 Alice : 125
可以看到的确如此。
下面,我们的语文老师希望以语文成绩排序,而数学老婆希望以数学成绩排序,英语老师则希望以英语成绩排序,……,班主任则希望以总分排序。
看到这里,你可以会说,我做一个方法来实现所有老师的愿望,这个方法有两个参数,一个是List对象,一个是type,type参数用来表示语文老师要的语文,数学老师要的数学等等。
这个方法当然是不错的,但不是最酷的。最酷的方法是语文老师调用sortByChinScore()方法,而数学老师调用sortByMathScore()方法,英语老师调用sortByEnglScore()方法,等等。
且慢!这不是要我写六七个方法来实现所有老师的要求?这未免有点无聊吧?像这样的代码写起来也枯燥乏味呀。
当然不用写六七个方法,答案就是“invokeMethod”方法。我们来看看是如何实现这样一个想法的:import java.util.Collectionsimport java.util.Comparator class SortHelper{ def list public SortHelper(list) { this.list = list } // 所有的以sort开头的方法都来调用“invokeMethod”,当然,其他方法也有可能来调用它,但我不做处理。 def invokeMethod(String name,Object args) { //首先判断方法名是否以“sortBy”开头,是则处理,否则不处理。 if(name.indexOf('sortBy')==0) { //取得属性名,如“ChinScore” name = name[6..name.length()-1] //把第一个字母由大写变小写,就取得了属性名 name = name[0].toLowerCase()+name[1..name.length()-1] //实现Comparator接口,大家可以参考jdk文档。 def comparator = { node1,node2 -> return node1."${name}".compareTo(node2."${name}") } as Comparator //排序 Collections.sort(this.list,comparator) } }}
真的很简单。下面我们来测试一些这个类:
def sorter = new SortHelper(scores) sorter.sortByChinScore() scores.each{ println it.name}
结果为:
Mike
Tom Alice
从语文成绩来看,Mike 88,Tom 90,Alice 100。排序是正确的。下面我们以数学成绩排序:
sorter.sortByMathScore() scores.each{ println it.name}
结果为:
Alice
Mike Tom
从数学成绩来看,Alice 55,Mike 90,Tom 99,排序也没有问题。大家可以再测几个看看。
像这样的功能的确够酷,我们今后在Groovy/Grails平台会经常遇到;同时,这样一个思想也可以发展成DSL的一种实现。
转载地址:https://lux-sun.blog.csdn.net/article/details/106150448 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!