This is an old revision of the document!


Visitor Pattern

객체 구조를 돌아다니면서 동일한 조작을 반복해서 적용하기. 이 때, '객체 구조'와 '처리(Algorithm)'을 분리하는 것이 핵심. 개방-폐쇄 원칙(OCP, Open-Closed Principle)1)을 적용한 방법중 하나.

  • 언제 사용하는가?
    • 데이터 구조 안에 많은 요소가 저장되어 있고, 그 각 요소에 대해서 다른 종류의 처리(Algorithm)가 필요한 경우
  • 시나리오
    • Composite Pattern의 예제인 “파일 시스템” 탐색을 이용. 여기서 처리 알고리즘을 분리시킨다.

Visitor

Method Overloading을 이용하여 각각의 처리할 Logic 수행하는 역할.

package visitor;

import element.objectstructure.concrete.Directory;
import element.objectstructure.concrete.File;

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

package visitor.concrete;

import java.util.Iterator;

import element.objectstructure.Entry;
import element.objectstructure.concrete.Directory;
import element.objectstructure.concrete.File;
import visitor.Visitor;

public class ListVisitor extends Visitor {

	private String currentDirectoryName = "";
	
	@Override
	public void visit(File file) {
		System.out.println(currentDirectoryName + " / " + file);
	}

	@Override
	public void visit(Directory directory) {
		System.out.println(currentDirectoryName + " / " + directory);
		
		String tempDirectoryName = currentDirectoryName;
		
		currentDirectoryName = new StringBuilder(currentDirectoryName).append("/").append(directory.getName()).toString();
		Iterator<Entry> it = directory.iterator();
		while(it.hasNext()) {
			it.next().accept(this);
		}
		
		currentDirectoryName = tempDirectoryName; 
	}
}

Element

방문할 객체를 지정하는 역할. accept()로 요청하여 객체 각각의 처리 Logic 수행.

package element;

import visitor.Visitor;

public interface Element {
	void accept(Visitor visitor);
}

package element.objectstructure;

import java.util.Iterator;

import element.Element;
import element.objectstructure.exception.FileTreatmentException;

public abstract class Entry implements Element {
	public abstract String getName();
	public abstract int getSize();
	
	/* Directory에서만 유효 */
	public Entry add(Entry entry) throws FileTreatmentException {
		throw new FileTreatmentException();
	}
	
	/* Directory에서만 유효 */
	public Iterator<Entry> iterator() throws FileTreatmentException {
		throw new FileTreatmentException();
	}
	
	@Override
	public String toString() {
		StringBuilder sb = new StringBuilder();
		sb.append(getName());
		sb.append(" (");
		sb.append(getSize());
		sb.append(") ");
		
		return sb.toString();
	}
}

package element.objectstructure.concrete;

import element.objectstructure.Entry;
import visitor.Visitor;

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 void accept(Visitor visitor) {
		visitor.visit(this);
	}

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

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

}

package element.objectstructure.concrete;

import java.util.ArrayList;
import java.util.Iterator;

import element.objectstructure.Entry;
import element.objectstructure.exception.FileTreatmentException;
import visitor.Visitor;

public class Directory extends Entry {

	private String name;
	private ArrayList<Entry> directories = new ArrayList<>();
	
	public Directory(String name) {
		this.name = name;
	}

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

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

	@Override
	public int getSize() {
		return directories.size();
	}
	
	@Override
	public Entry add(Entry entry) throws FileTreatmentException {
		directories.add(entry);
		return this;
	}
	
	@Override
	public Iterator<Entry> iterator() throws FileTreatmentException {
		return directories.iterator();
	}
}

design_pattern/visitor_pattern.1509440221.txt.gz · Last modified: 2021/02/07 03:15 (external edit)