TypeLiteral.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.lang3.reflect;

  18. import java.lang.reflect.Type;
  19. import java.lang.reflect.TypeVariable;

  20. import org.apache.commons.lang3.Validate;

  21. /**
  22.  * Type literal comparable to {@code javax.enterprise.util.TypeLiteral},
  23.  * made generally available outside the JEE context. Allows the passing around of
  24.  * a "token" that represents a type in a typesafe manner, as opposed to
  25.  * passing the (non-parameterized) {@link Type} object itself. Consider:
  26.  * <p>
  27.  * You might see such a typesafe API as:
  28.  * <pre>{@code
  29.  * class Typesafe {
  30.  *   <T> T obtain(Class<T> type, ...);
  31.  * }
  32.  * }
  33.  * </pre>
  34.  * Consumed in the manner of:
  35.  * <pre>
  36.  * Foo foo = typesafe.obtain(Foo.class, ...);
  37.  * </pre>
  38.  * Yet, you run into problems when you want to do this with a parameterized type:
  39.  * <pre>{@code
  40.  * List<String> listOfString = typesafe.obtain(List.class, ...); // could only give us a raw List
  41.  * }</pre>
  42.  * {@link java.lang.reflect.Type} might provide some value:
  43.  * <pre>{@code
  44.  * Type listOfStringType = ...; // firstly, how to obtain this? Doable, but not straightforward.
  45.  * List<String> listOfString = (List<String>) typesafe.obtain(listOfStringType, ...); // nongeneric Type would necessitate a cast
  46.  * }</pre>
  47.  * The "type literal" concept was introduced to provide an alternative, i.e.:
  48.  * <pre>{@code
  49.  * class Typesafe {
  50.  *   <T> T obtain(TypeLiteral<T> type, ...);
  51.  * }
  52.  * }</pre>
  53.  * Consuming code looks like:
  54.  * <pre>{@code
  55.  * List<String> listOfString = typesafe.obtain(new TypeLiteral<List<String>>() {}, ...);
  56.  * }</pre>
  57.  * <p>
  58.  * This has the effect of "jumping up" a level to tie a {@link java.lang.reflect.Type}
  59.  * to a type variable while simultaneously making it short work to obtain a
  60.  * {@link Type} instance for any given type, inline.
  61.  * </p>
  62.  * <p>Additionally {@link TypeLiteral} implements the {@link Typed} interface which
  63.  * is a generalization of this concept, and which may be implemented in custom classes.
  64.  * It is suggested that APIs be defined in terms of the interface, in the following manner:
  65.  * </p>
  66.  * <pre>{@code
  67.  *   <T> T obtain(Typed<T> typed, ...);
  68.  * }</pre>
  69.  *
  70.  * @param <T> the type
  71.  * @since 3.2
  72.  */
  73. public abstract class TypeLiteral<T> implements Typed<T> {

  74.     @SuppressWarnings("rawtypes")
  75.     private static final TypeVariable<Class<TypeLiteral>> T = TypeLiteral.class.getTypeParameters()[0];

  76.     /**
  77.      * Represented type.
  78.      */
  79.     public final Type value;

  80.     private final String toString;

  81.     /**
  82.      * The default constructor.
  83.      */
  84.     protected TypeLiteral() {
  85.         this.value =
  86.             Validate.notNull(TypeUtils.getTypeArguments(getClass(), TypeLiteral.class).get(T),
  87.                 "%s does not assign type parameter %s", getClass(), TypeUtils.toLongString(T));

  88.         this.toString = String.format("%s<%s>", TypeLiteral.class.getSimpleName(), TypeUtils.toString(value));
  89.     }

  90.     @Override
  91.     public final boolean equals(final Object obj) {
  92.         if (obj == this) {
  93.             return true;
  94.         }
  95.         if (!(obj instanceof TypeLiteral)) {
  96.             return false;
  97.         }
  98.         final TypeLiteral<?> other = (TypeLiteral<?>) obj;
  99.         return TypeUtils.equals(value, other.value);
  100.     }

  101.     @Override
  102.     public Type getType() {
  103.         return value;
  104.     }

  105.     @Override
  106.     public int hashCode() {
  107.         return 37 << 4 | value.hashCode();
  108.     }

  109.     @Override
  110.     public String toString() {
  111.         return toString;
  112.     }
  113. }