![JAR search and dependency download from the Maven repository](/logo.png)
org.apache.juneau.annotation.AnnotationImpl Maven / Gradle / Ivy
// ***************************************************************************************************************************
// * Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file *
// * distributed with this work for additional information regarding copyright ownership. The ASF licenses this file *
// * to you under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance *
// * with the License. You may obtain a copy of the License at *
// * *
// * http://www.apache.org/licenses/LICENSE-2.0 *
// * *
// * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an *
// * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the *
// * specific language governing permissions and limitations under the License. *
// ***************************************************************************************************************************
package org.apache.juneau.annotation;
import static org.apache.juneau.collections.JsonMap.*;
import static org.apache.juneau.common.internal.ThrowableUtils.*;
import static java.util.Arrays.*;
import java.lang.annotation.*;
import java.lang.reflect.*;
import java.util.*;
import org.apache.juneau.collections.*;
import org.apache.juneau.internal.*;
/**
* A concrete implementation of an annotation.
*
*
* Follows the standard Java conventions for equality and hashcode calculation for annotations.
* Equivalent annotations defined programmatically and declaratively should match for equality and hashcode calculation.
*
*
* For performance reasons, the hashcode is calculated one time and cached at the end of object creation.
* Constructors must call the {@link #postConstruct()} method after all fields have been set to trigger this calculation.
*
*
See Also:
*
*/
public class AnnotationImpl implements Annotation {
private final Class extends Annotation> annotationType;
private int hashCode = -1;
/**
* Constructor.
*
* @param b The builder used to instantiate the fields of this class.
*/
public AnnotationImpl(AnnotationBuilder b) {
this.annotationType = b.annotationType;
}
/**
* This method must be called at the end of initialization to calculate the hashCode one time.
*/
protected void postConstruct() {
this.hashCode = AnnotationUtils.hashCode(this);
}
/**
* Implements the {@link Annotation#annotationType()} method for child classes.
*
* @return This class.
*/
@Override /* Annotation */
public Class extends Annotation> annotationType() {
return annotationType;
}
@Override /* Object */
public int hashCode() {
if (hashCode == -1)
throw new IllegalArgumentException("Programming error. postConstruct() was never called on annotation.");
return hashCode;
}
@Override /* Object */
public boolean equals(Object o) {
if (! annotationType.isInstance(o))
return false;
return AnnotationUtils.equals(this, (Annotation)o);
}
/**
* Returns this annotation as a map of key/value pairs.
*
*
* Useful for debugging.
*
* @return This annotation as a map of key/value pairs.
*/
public JsonMap toMap() {
JsonMap m = create();
stream(annotationType().getDeclaredMethods())
.filter(x->x.getParameterCount() == 0 && x.getDeclaringClass().isAnnotation())
.sorted(Comparator.comparing(Method::getName))
.forEach(x -> m.append(x.getName(), safeSupplier(()->x.invoke(this))));
return m;
}
@Override /* Object */
public String toString() {
return toMap().asString();
}
}