站点图标 AI技术聚合

Java中的functor实现

  经常听到回调函数(callback function)这个概念, 所谓回调函数,就是指这个函数先在某处注册,而它将在稍后某个需要的时候被调用。比如在利用SDK 进行Windows编程的时候,我们需要注册一个WNDCLASS类,这个类中有这样一个参数 lpfnWndProc, 要进行消息处理,我们就要用处理消息的函数的指针给它赋值。消息处理函数什么时候被调用的?我们没有显式地在程序中看到啊。是OS调用的。  这是SDK的试验方式,当然用的是过程式的语言C,可以通过传递函数的指针实现。

   C++中怎么来实现呢?当然,C++兼容C,用函数指针就可以。  同时C++又提供了面向对象的机制,可不可以有不同的实现机制呢?  当然! STL 中的functor(Function object)就可以用到回调上。  比如对一个存放int数据的vector进行递减排序的话,我们可以这样进行。

  sort(vec.begin(),vec.end(),greater<int>());
   greater<int>()
  

   就是我们传递的一个匿名对象,它重载了函数调用运算符“()”。我们没有显式地调用这个对象里面提供的函数,sort函数对对象里面的函数进行call back。

  Java中要实现类似functor的功能,应该怎么办呢?Command模式可以帮上忙。Command模式看起来很简单,只要把command封装到一个接口中就可以。Command模式是回调机制的一个面向对象的替代品。

  比如 java.io 中已经定义好的一个接口

      public interface FilenameFilter {
          boolean accept(File dir, String name);
  
    }

  这个FilenameFilter就是Command,实现Command的类就是ConcreteCommand。这个接口所声明的操作 “accept” 就是看看目录dir中的文件name是否满足某种要求,如果满足就返回true,否则就返回false。这个要求是什么呢?你要对这个接口进行实现。比如我想看看这个文件的名称包含不包含指定的字符串,那么就可以定义下面的类:
   class DirFilter implements FilenameFilter {
        private String afn;
        public DirFilter(String afn){
            this.afn = afn;
        }  
        public boolean accept(File dir, String name){
            String f = new File(name).getName();
            return f.indexOf(afn) != -1;
        }
    }

   怎么样使用它呢?File类中有这样一个方法
      public String[] list(FilenameFilter filter)

  因此,我们就可以这样做了:
      File file = new File(“.”);
      String[] list = file.list(new DirFilter(“wf”));

  得到的list就是一个当然目录中所有包含字符串”wf”的文件名称的字符串数组。怎么样,看起来是不是和C++中的functor差不多呢?

退出移动版