假设我们想查看一个目录列表,可以用两种方法列出文件对象。如果我们调用不带参数的
list()方法,便可以获得此文件对象包含的全部列表。然而,如果我们想获得一个受限列
表——例如,想得到所有扩展名为.java 的文件——那么我们就要用到“目录过滤器”,这个
类会告诉我们怎样显示符合条件的文件对象。
下面给出一个例子的代码,注意:通过使用 java.utils.Arrays.sort( )和第 11 章中定义的
AlphabeticComparator,可以很容易地对结果进行排序(按字母顺序)。
//: c12:DirList.java
// Displays directory listing using regular expressions.
// {Args: "D.*\.java"}
import java.io.*;
import java.util.*;
import java.util.regex.*;
import com.bruceeckel.util.*;
public class DirList {
public static void main(String[] args) {
File path = new File(".");
String[] list;
if(args.length == 0)
list = path.list();
else
list = path.list(new DirFilter(args[0]));
Arrays.sort(list, new AlphabeticComparator());
for(int i = 0; i < list.length; i++)
System.out.println(list[i]);
}
}
class DirFilter implements FilenameFilter {
private Pattern pattern;
public DirFilter(String regex) {
pattern = Pattern.compile(regex);
}
public boolean accept(File dir, String name) {
// Strip path information, search for regex:
return pattern.matcher(
new File(name).getName()).matches();
}
} ///:~
这里,DirFilter 类“实现”了 FilenameFilter 接口。有必要先看看 FilenameFilter 接口是
多么的简单:
public interface FilenameFilter {
boolean accept(File dir, String name);
}
这表示此种类型的对象一定要提供一个 accept()方法。创建这个类的目的在于把 accept
()方法提供给 list()使用,使 list()可以回调 accept()进而以决定哪些文件包含在
列表中。因此,这种结构也常常称为“回调(call back)”。更具体地说,这是一个策略模
式的例子,因为 list()实现了基本的功能,而且我们按照 FilenameFilter 的形式提供了
这个策略,以便完善 list()在提供服务时所需的算法。因为 list()接受 FilenameFilter
对象作为参数,这意味着我们可以传递实现了 FilenameFilter 接口的任何对象,用以选择
(甚至在运行时)list()方法的行为。回调的目的就是提供了代码行为的灵活性。
DirFilter 说明:正因为一个接口仅是一组方法集,所以我们也就没有被限定为只能编写那
些方法。(然而,在一个接口中,我们必须为其所有方法提供定义。)在这种情形下,也就可
以创建 DirFilter 的构造器了。
accept()方法必须接受一个文件对象,此对象代表某个特定文件所在目录以及包含那个
文件名的一个字符串。我们可以选择使用或忽视这些参数,但是至少会用到那个文件的名字。
记住一点:list()方法会为此目录对象下的每个文件名调用 accept(),来判断该文件是
否包含在内;判断结果由 accept()返回的布尔值表示。
为了确定所处理的元素仅有文件名并且不包含路径信息,我们必须持有一个 String 对象,
并在其外创建一个文件对象,然后调用 getName(),该方法剔除掉所有的路径信息(以
平台独立的方式)。接着 accept()会使用一个正则表达式的 matcher 对象,来查看此
正则表达式 regex 是否匹配这个文件的名字。通过使用 accept(),list()方法会返回
list()方法,便可以获得此文件对象包含的全部列表。然而,如果我们想获得一个受限列
表——例如,想得到所有扩展名为.java 的文件——那么我们就要用到“目录过滤器”,这个
类会告诉我们怎样显示符合条件的文件对象。
下面给出一个例子的代码,注意:通过使用 java.utils.Arrays.sort( )和第 11 章中定义的
AlphabeticComparator,可以很容易地对结果进行排序(按字母顺序)。
//: c12:DirList.java
// Displays directory listing using regular expressions.
// {Args: "D.*\.java"}
import java.io.*;
import java.util.*;
import java.util.regex.*;
import com.bruceeckel.util.*;
public class DirList {
public static void main(String[] args) {
File path = new File(".");
String[] list;
if(args.length == 0)
list = path.list();
else
list = path.list(new DirFilter(args[0]));
Arrays.sort(list, new AlphabeticComparator());
for(int i = 0; i < list.length; i++)
System.out.println(list[i]);
}
}
class DirFilter implements FilenameFilter {
private Pattern pattern;
public DirFilter(String regex) {
pattern = Pattern.compile(regex);
}
public boolean accept(File dir, String name) {
// Strip path information, search for regex:
return pattern.matcher(
new File(name).getName()).matches();
}
} ///:~
这里,DirFilter 类“实现”了 FilenameFilter 接口。有必要先看看 FilenameFilter 接口是
多么的简单:
public interface FilenameFilter {
boolean accept(File dir, String name);
}
这表示此种类型的对象一定要提供一个 accept()方法。创建这个类的目的在于把 accept
()方法提供给 list()使用,使 list()可以回调 accept()进而以决定哪些文件包含在
列表中。因此,这种结构也常常称为“回调(call back)”。更具体地说,这是一个策略模
式的例子,因为 list()实现了基本的功能,而且我们按照 FilenameFilter 的形式提供了
这个策略,以便完善 list()在提供服务时所需的算法。因为 list()接受 FilenameFilter
对象作为参数,这意味着我们可以传递实现了 FilenameFilter 接口的任何对象,用以选择
(甚至在运行时)list()方法的行为。回调的目的就是提供了代码行为的灵活性。
DirFilter 说明:正因为一个接口仅是一组方法集,所以我们也就没有被限定为只能编写那
些方法。(然而,在一个接口中,我们必须为其所有方法提供定义。)在这种情形下,也就可
以创建 DirFilter 的构造器了。
accept()方法必须接受一个文件对象,此对象代表某个特定文件所在目录以及包含那个
文件名的一个字符串。我们可以选择使用或忽视这些参数,但是至少会用到那个文件的名字。
记住一点:list()方法会为此目录对象下的每个文件名调用 accept(),来判断该文件是
否包含在内;判断结果由 accept()返回的布尔值表示。
为了确定所处理的元素仅有文件名并且不包含路径信息,我们必须持有一个 String 对象,
并在其外创建一个文件对象,然后调用 getName(),该方法剔除掉所有的路径信息(以
平台独立的方式)。接着 accept()会使用一个正则表达式的 matcher 对象,来查看此
正则表达式 regex 是否匹配这个文件的名字。通过使用 accept(),list()方法会返回
一个数组。