JavaVisitor模式

本文最后更新于:1 年前

访问数据结构并处理数据

1 - 表示访问者的抽象类, 它访问文件和文件夹

public abstract class Visitor {
    public abstract void visit(File file);
    public abstract void visit(Directory directory);
}

2 - 表示数据结构的接口, 它接受访问者的访问

public interface Element {
    public abstract void accept(Visitor v);
}

3 - Visitor类的子类, 显示文件和文件夹一览

public class ListVisitor extends Visitor{
    private String currentDir = "";

    @Override
    public void visit(File file) {
        System.out.println(currentDir + "/" + file);
    }

    @Override
    public void visit(Directory directory) {
        System.out.println(currentDir + "/" + directory);
        String saveDir = currentDir;
        currentDir = currentDir + "/" + directory.getName();
        Iterator it = directory.iterator();
        while (it.hasNext()){
            Entry entry = (Entry)it.next();
            entry.accept(this);
        }
        currentDir = saveDir;
    }
}

4 - 抽象类, File类和Directory类的父类, 实现了Element接口

public abstract class Entry implements Element{
    public abstract String getName();   // 获取名字
    public abstract int getSize();      // 获取大小
    public Entry add(Entry entry) throws FileTreatMeantException { // 加入条目目录
        throw new FileTreatMeantException();
    }

    public Iterator iterator() throws FileTreatMeantException { // 生成Iterator
        throw new FileTreatMeantException();
    }
                                                        // 显示条目一览
    @Override                                           // 显示代表类的文字
    public String toString(){
        return getName() + " (" + getSize() + ")";
    }
}

5 - 表示文件的类

public class File extends Entry{
    private String name;
    private int size;

    public File(String name, int size){
        this.name = name;
        this.size = size;
    }

    @Override
    public String getName() {
        return name;
    }

    @Override
    public int getSize() {
        return size;
    }

    @Override
    public void accept(Visitor v) {
        v.visit(this);
    }
}

6 - 表示文件夹的类

public class Directory extends Entry{
    private String name;
    private ArrayList directory = new ArrayList();
    public Directory(String name){
        this.name = name;
    }

    @Override
    public String getName() {
        return name;
    }

    @Override
    public int getSize() {
        int size = 0;
        for (Object o : directory) {
            Entry entry = (Entry) o;
            size += entry.getSize();
        }
        return size;
    }

    public Entry add(Entry entry){
        directory.add(entry);
        return this;
    }

    public Iterator iterator(){
        return directory.iterator();
    }

    @Override
    public void accept(Visitor v) {
        v.visit(this);
    }
}

7 - 表示向文件中增加Entry时发生的异常的类

public class FileTreatMeantException extends RuntimeException{

    public FileTreatMeantException(){
    }

    public FileTreatMeantException(String msg){
        super(msg);
    }
}

8 - 测试程序行为的类

public class Main {
    public static void main(String[] args) {
        try {
            System.out.println("Making root entries...");
            Directory rootDir = new Directory("root");
            Directory binDir = new Directory("bin");
            Directory tmpDir = new Directory("tmp");
            Directory usrDir = new Directory("usr");
            rootDir.add(binDir);
            rootDir.add(tmpDir);
            rootDir.add(usrDir);
            binDir.add(new File("vi", 10000));
            binDir.add(new File("latex", 20000));
            rootDir.accept(new ListVisitor());

            System.out.println("");
            System.out.println("Making user entries...");
            Directory yuki = new Directory("yuki");
            Directory hanako = new Directory("hanako");
            Directory tomura = new Directory("tomura");
            usrDir.add(yuki);
            usrDir.add(hanako);
            usrDir.add(tomura);
            yuki.add(new File("diary.html", 100));
            yuki.add(new File("Composite.java", 200));
            hanako.add(new File("memo.tex", 300));
            tomura.add(new File("game.doc", 400));
            tomura.add(new File("junk.mail", 500));
            rootDir.accept(new ListVisitor());

        }catch (FileTreatMeantException e){
            e.printStackTrace();
        }
    }
}

9 - 输出样例

Making root entries...
/root (30000)
/root/bin (30000)
/root/bin/vi (10000)
/root/bin/latex (20000)
/root/tmp (0)
/root/usr (0)

Making user entries...
/root (31500)
/root/bin (30000)
/root/bin/vi (10000)
/root/bin/latex (20000)
/root/tmp (0)
/root/usr (1500)
/root/usr/yuki (300)
/root/usr/yuki/diary.html (100)
/root/usr/yuki/Composite.java (200)
/root/usr/hanako (300)
/root/usr/hanako/memo.tex (300)
/root/usr/tomura (900)
/root/usr/tomura/game.doc (400)
/root/usr/tomura/junk.mail (500)

本博客所有文章除特别声明外,均采用 CC BY-SA 4.0 协议 ,转载请注明出处!