반응형

javax.lang.model.util.AbstractAnnotationValueVisitor7는 자바 언어에서 사용되는 Annotation Processor를 개발할 때 유용한 클래스 중 하나입니다. 이 클래스는 javax.lang.model.element.AnnotationValueVisitor 인터페이스를 구현하는 추상 클래스로, 자바 언어에서 정의된 어노테이션 값을 처리하는 데 사용됩니다.

AbstractAnnotationValueVisitor7은 추상 클래스이므로 이 클래스를 직접 사용하는 것은 불가능합니다. 대신 이 클래스를 상속받아 자신만의 어노테이션 값을 처리하는 클래스를 만들어 사용해야 합니다.

이 클래스는 자바 7에서 추가된 AnnotationValueVisitor의 버전 7에 해당합니다. AnnotationValueVisitor는 어노테이션의 값에 대해 visit 메서드를 정의해야 합니다. AbstractAnnotationValueVisitor7은 이러한 visit 메서드를 모두 구현하고 있으며, 이를 상속받은 클래스는 필요한 visit 메서드만 재정의하여 사용할 수 있습니다.

AbstractAnnotationValueVisitor7 클래스의 주요 메서드는 다음과 같습니다.

1. visit(AnnotationValue av, P p): 어노테이션 값을 처리하기 위한 기본 메서드입니다. 이 메서드는 AnnotationValueVisitor 인터페이스에서 정의되며, 어노테이션의 값에 대한 visit 메서드를 호출합니다.

2. defaultAction(Object o, P p): 어노테이션 값이 특별한 유형이 아닐 경우 호출되는 기본 메서드입니다.

3. visitBoolean(boolean b, P p): boolean 유형의 어노테이션 값을 처리하는 메서드입니다.

4. visitByte(byte b, P p): byte 유형의 어노테이션 값을 처리하는 메서드입니다.

5. visitChar(char c, P p): char 유형의 어노테이션 값을 처리하는 메서드입니다.

6. visitDouble(double d, P p): double 유형의 어노테이션 값을 처리하는 메서드입니다.

7. visitFloat(float f, P p): float 유형의 어노테이션 값을 처리하는 메서드입니다.

8. visitInt(int i, P p): int 유형의 어노테이션 값을 처리하는 메서드입니다.

9. visitLong(long l, P p): long 유형의 어노테이션 값을 처리하는 메서드입니다.

10. visitShort(short s, P p): short 유형의 어노테이션 값을 처리하는 메서드입니다.

11. visitString(String s, P p): String 유형의 어노테이션 값을 처리하는 메서드입니다.

12. visitType(TypeMirror t, P p): TypeMirror 유형의 어노테이션 값을 처리하는 메서드입니다.

13. visitEnumConstant(VariableElement c, P p): Enum 유형의 어노테이션 값을 처리하는 메서드입니다.

14. visitAnnotation(Annotation14. visitAnnotation(AnnotationMirror a, P p): 다른 어노테이션 유형의 어노테이션 값을 처리하는 메서드입니다.

15. visitArray(List<? extends AnnotationValue> vals, P p): 배열 유형의 어노테이션 값을 처리하는 메서드입니다.

AbstractAnnotationValueVisitor7 클래스는 이 외에도 다양한 visit 메서드를 지원하며, 이를 사용해 자신만의 어노테이션 값을 처리하는 메서드를 만들 수 있습니다. 이 클래스를 상속받아 사용하는 경우, AnnotationValueVisitor 인터페이스에서 정의된 visit 메서드를 모두 구현해야 합니다.

예를 들어, 다음은 AbstractAnnotationValueVisitor7를 상속받아 자신만의 어노테이션 값을 처리하는 클래스를 만드는 예시입니다.

 

 

import javax.lang.model.element.AnnotationValue;
import javax.lang.model.element.AnnotationValueVisitor;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.AbstractAnnotationValueVisitor7;

public class MyAnnotationValueVisitor extends AbstractAnnotationValueVisitor7<String, Void> {
    @Override
    public String visitString(String s, Void aVoid) {
        return s.toUpperCase();
    }

    @Override
    public String visitType(TypeMirror t, Void aVoid) {
        return t.toString().toLowerCase();
    }

    @Override
    public String visit(AnnotationValue av, Void aVoid) {
        return av.accept(this, null);
    }
}

이 클래스에서는 visitString, visitType, visit 메서드를 재정의하여 각각 String, TypeMirror, AnnotationValue 유형의 어노테이션 값을 처리하는 메서드를 만들었습니다. MyAnnotationValueVisitor 클래스를 사용하면 어노테이션 값을 처리하는 데 필요한 visit 메서드를 모두 구현할 필요 없이, 필요한 visit 메서드만 재정의하여 사용할 수 있습니다.각 visit 메서드는 어노테이션 값을 처리하는 데 필요한 코드를 구현해야 합니다. 아래는 예시 코드입니다.

 

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.util.List;
import javax.lang.model.element.AnnotationValue;
import javax.lang.model.element.AnnotationValueVisitor;
import javax.lang.model.element.Element;
import javax.lang.model.element.TypeElement;
import javax.lang.model.element.VariableElement;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.AbstractAnnotationValueVisitor7;
import javax.tools.Diagnostic.Kind;
import javax.annotation.processing.ProcessingEnvironment;
import javax.annotation.processing.AbstractProcessor;
import javax.annotation.processing.RoundEnvironment;
import javax.annotation.processing.SupportedAnnotationTypes;

@SupportedAnnotationTypes("MyAnnotation")
public class MyProcessor extends AbstractProcessor {

  @Override
  public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
    for (TypeElement annotation : annotations) {
      for (Element element : roundEnv.getElementsAnnotatedWith(annotation)) {
        if (element.getKind() == ElementKind.FIELD) {
          VariableElement variableElement = (VariableElement) element;
          MyAnnotation myAnnotation = variableElement.getAnnotation(MyAnnotation.class);
          if (myAnnotation != null) {
            MyAnnotationVisitor visitor = new MyAnnotationVisitor(processingEnv);
            AnnotationValue value = myAnnotation.value();
            value.accept(visitor, variableElement);
          }
        }
      }
    }
    return true;
  }

  private static class MyAnnotationVisitor extends AbstractAnnotationValueVisitor7<Void, Element> {

    private final ProcessingEnvironment processingEnv;

    public MyAnnotationVisitor(ProcessingEnvironment processingEnv) {
      this.processingEnv = processingEnv;
    }

    @Override
    public Void visitString(String value, Element element) {
      if (!value.matches("\\w+")) {
        processingEnv.getMessager().printMessage(Kind.ERROR, "Invalid value: " + value, element);
      }
      return null;
    }

    @Override
    public Void visitEnumConstant(VariableElement c, Element element) {
      TypeElement typeElement = (TypeElement) c.getEnclosingElement();
      String typeName = typeElement.getQualifiedName().toString();
      String constantName = c.getSimpleName().toString();
      processingEnv.getMessager().printMessage(Kind.NOTE, "Enum constant: " + typeName + "." + constantName, element);
      return null;
    }

    @Override
    public Void visitArray(List<? extends AnnotationValue> values, Element element) {
      for (AnnotationValue value : values) {
        value.accept(this, element);
      }
      return null;
    }

    // other visit methods

  }
}

위 예제 코드는 MyAnnotation 이라는 어노테이션을 처리하는 프로세서를 작성한 것입니다. MyAnnotation은 String value()와 ElementType type()이라는 두 개의 속성을 가지고 있습니다. 이 예제에서는 value() 속성의 값을 처리하는 visitString 메서드와 type() 속성의 값을 처리하는 visitEnumConstant 메서드를 구현하였습니다. 또한, visitArray 메서드를 이용하여 배열 타입의 값을 처리하는 visit 메서드도 함께 구현하였습니다.

 

반응형

+ Recent posts