Groovy探索之MOP 十五 方法名的动态性(1)

         Groovy探索之MOP 十五  方法名的动态性(1

 

 

 

到目前为止,我们的《Groovy探索之MOP》系列已经谈到了使用ExpandoMetaClass的方方面面,但值得注意的是,我们通过ExpandoMetaClass给一个类在运行期内添加一个方法,不管是普通方法还是静态方法,我们都是添加一个确定方法名的方法。即我们添加一个方法名为A的方法,然后才能使用这个方法A

然而,方法名的动态性,其实是我们早已接触过的事情,比如在《Groovy探索之invokeMethod方法》里,我们就可以创建形如“sortByXxxxx()”这样漂亮的方法。当然了,在那篇文字里,我们是通过“invokeMethod”方法来实现的。

我们都知道,使用ExpandoMetaClass在运行期内给一个对象或类添加一个方法要比hook方法简单得多,所以,我们在实际的编码过程中,更多时候会用到ExpandoMetaClass来实现。那么,我们使用ExpandoMetaClass是否也可以实现上面hook方法所实现了的方法名的动态性呢?

我们知道,使用ExpandoMetaClass在运行期内添加一个方法(不管是给对象还是给类),形式都是像下面的样子:

 

类名.metaClass.方法名 = 方法体

 

同时,上面的公式还可以写成如下的样子:

 

 

类名.metaClass. "方法名" = 方法体

 

这下我们明白了,既然方法名可以用双引号括起来,那么,如果双引号里面是一个Gstring对象的话,这样,我们就可以给方法赋予一个可变的数据了。从而达到了我们使用ExpandoMetaClass实现方法名的动态性的目的了。

 

废话少说,我们还是先来看一个例子吧!

 

现在,我们有一个T类,如下:

 

class T

{

   

}

 

 

我们就可以这样对T类添加方法:

 

      String functionName = "a";

     

      T.metaClass."$functionName" = {

             println "invoke $functionName"

      }

     

      def t = new T()

     

      t.a()

   

 

我们在添加方法的时候,使用了一个Gstring对象来代替了写死的字符串,这样就实现了方法名的动态性。运行结果为:

 

invoke a

 

当然了,你可能会觉得上面的方法名的动态性还不够清晰,那么,我们可以实现如下的一个静态方法,来达到我们方法名动态性的目的:

 

    def static add(functionName)

    {

       T.metaClass."$functionName" = {

                println "invoke $functionName"

         }

    }

   

 

这样,我们就可以如下来使用它了:

 

      add('b')

     

      def t1 = new T()

     

      t1.b()

     

     

      add('c')

     

      def t2 = new T()

     

      t2.c()

   

 

运行结果为:

invoke b

invoke c

 

上面的例子只是小儿科,我们使用ExpandoMetaClass实现的方法名的动态性同样可以使用于《Groovy探索之invokeMethod方法》中的那个有名的例子。关于那个例子的来历,大家可以在原文中看到,这里就不再重述。

在那篇文字中,Student类是这样的:

class Student {

   

    String no;

    String name;

    float chinScore;

    float mathScore;

    float englScore;

    float physScore;

    float chemScore;

    float totalScore;

 

}

 

 

同样,我们的那个排序工具类还是保留,只是不再实现它的排序方法了,如下:

 

class SortHelper

{

 

    def list

   

    public SortHelper(list)

    {

       this.list = list

    }

 

}

 

 

当然,我们的排序方法是要留给ExpandoMetaClass来实现的,下面就是:

 

      ['chinScore','mathScore','englScore','physScore','chemScore','totalScore'].each{

         name ->

           def name1 = name[0].toUpperCase()+name[1..-1]

           SortHelper.metaClass."sortBy${name1}" = {

               -> 

                  def comparator = {

                         node1,node2 ->

                         return node1."${name}".compareTo(node2."${name}")

                  } as Comparator

                 

                  Collections.sort(delegate.list,comparator)

           }

      }

 

 

 

我们的想法就是,把要实现的方法的方法名(或部分)组成一个数组,然后依次遍历,交给ExpandoMetaClass来添加方法。

上面的代码都很简单,在这里我们就不再多说了。

最后,我们来写代码测试上面的添加方法了。如下:

 

 

 

你可能感兴趣的:(String,list,Class,groovy,float,hook)