所谓回调函数:就是当我们在设计一个通用算法或函数时,其中的一部分细节需要调用者提供。而该部分由调用者实现的功能,在我实现通用算法或者函数时,假想该部分功能已经有一个函数实现,我知道该假想函数的调用规范,并且我们将指向该假想函数的指针(c中,)或者函数对象,做为该通用函数的一个参数。该假想函数,称为回调函数,有用户调用该通用函数时提供。
好像设计好一个功能框架,回调函数是一个按照该功能框架的某个功能规范的要求的实现的具体函数;该函数由调用者实现,但是要符合框架功能的规范。
个人认为,之所以称之为回调函数,是因为当我们调用这个通用算法时,这个通用算法要回过头来调用我们实现的具体功能函数(回调函数)。
而不同的调用者可以实现不同的(符合规范)回调函数,从而那个通用函数就可以实现各种各样的功能,这是将其称之为通用函数的缘故吧。
典型的例子,sort(),排序函数。 c中qsort(),C++中STL中Algorithm中,sort。 STL中很多通用算法的设计,大概就是基于回调函数的这种设计方法吧。(个人理解)
这种通用算法的设计的一个好处,就是有人常说的,避免重新发明轮子。。
看一下转载的这个帖子:
讲到回调函数这个概念给大家普及下, 所谓回调函数,就是指这个函数先在某处注册,而它将在稍后某个需要的时候被调用。比如在利用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{ public boolean accept(File dir, String name); }
这个FilenameFilter就是Command,实现Command的类就是ConcreteCommand.
这个接口所声明的操作"accept" 就是看看目录dir中的文件name是否满足某种要求,
如果满足就返回true,否则就返回false.
这个要求是什么呢?你要对这个接口进行实现。
比如我想看看这个文件的名称包含不包含指定的字符串,那么就可以定义下面的类:
哎,提供的例子愣是没看懂,查了下资料,自己实现了两个filenamefilter: FileNameFilter.java 和FileTypeFilter.java
<1> FileNameFilter.java
package cn.wzb; import java.io.File; import java.io.FilenameFilter; public class FileNameFilter implements FilenameFilter { private String filter; public FileNameFilter(String filter) { this.filter = filter; } @Override public boolean accept(File dir, String fileName) { return fileName.contains(filter); } }
<2>FileTypeFilter.java
package cn.wzb; import java.io.File; import java.io.FilenameFilter; public class FileTypeFilter implements FilenameFilter{ private String typeFilter; public FileTypeFilter(String typeFilter){ this.typeFilter = typeFilter; } @Override public boolean accept(File dir, String name) { return name.endsWith(this.typeFilter); } }
<3> 一个小的测试驱动程序:TestFileFilter.java
package cn.wzb; import java.io.File; public class TestFileFilter { public static void main(String[] args) { File dir = new File("E:/wzb"); String[] fileNames; System.out.println("FileNameFilter: "); fileNames = dir.list(new FileNameFilter("s")); //显示dir目录中,文件名(含目录)中包含s的文件名 for (String name : fileNames) { System.out.println(name); } System.out.println(); System.out.println("FileTypeFilter:"); fileNames = dir.list(new FileTypeFilter(".txt")); //显示dir目录中的txt文件 for (String fileName : fileNames) { System.out.println(fileName); } } }
发现C系列的语言,有很多共通之处,也许这是算法或者通用设计模式的一致性。 个人非常同意的一个观点就是语言无优劣,.....。关键是看谁在用。
有哲学的话,一直觉得非常有意思:特性包含共性。在编程语言方面,大概就是说,各种语言在一定程度上是有共同性的。