E
E
Ernesto Guevara2015-09-13 23:54:56
Java
Ernesto Guevara, 2015-09-13 23:54:56

How to get generic type information in Java at runtime?

I am working with the jackson library for Android and ran into a misunderstanding of working with Generics through reflection.
The library has a crutch TypeReference:

package com.fasterxml.jackson.core.type;

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;

/**
 * This generic abstract class is used for obtaining full generics type information
 * by sub-classing; it must be converted to {@link ResolvedType} implementation
 * (implemented by <code>JavaType</code> from "databind" bundle) to be used.
 * Class is based on ideas from
 * <a href="http://gafter.blogspot.com/2006/12/super-type-tokens.html"
 * >http://gafter.blogspot.com/2006/12/super-type-tokens.html</a>,
 * Additional idea (from a suggestion made in comments of the article)
 * is to require bogus implementation of <code>Comparable</code>
 * (any such generic interface would do, as long as it forces a method
 * with generic type to be implemented).
 * to ensure that a Type argument is indeed given.
 *<p>
 * Usage is by sub-classing: here is one way to instantiate reference
 * to generic type <code>List&lt;Integer></code>:
 *<pre>
 *  TypeReference ref = new TypeReference&lt;List&lt;Integer>>() { };
 *</pre>
 * which can be passed to methods that accept TypeReference, or resolved
 * using <code>TypeFactory</code> to obtain {@link ResolvedType}.
 */
public abstract class TypeReference<T> implements Comparable<TypeReference<T>>
{
    protected final Type _type;
    
    protected TypeReference()
    {
        Type superClass = getClass().getGenericSuperclass();
        if (superClass instanceof Class<?>) { // sanity check, should never happen
            throw new IllegalArgumentException("Internal error: TypeReference constructed without actual type information");
        }
        /* 22-Dec-2008, tatu: Not sure if this case is safe -- I suspect
         *   it is possible to make it fail?
         *   But let's deal with specific
         *   case when we know an actual use case, and thereby suitable
         *   workarounds for valid case(s) and/or error to throw
         *   on invalid one(s).
         */
        _type = ((ParameterizedType) superClass).getActualTypeArguments()[0];
    }

    public Type getType() { return _type; }
    
    /**
     * The only reason we define this method (and require implementation
     * of <code>Comparable</code>) is to prevent constructing a
     * reference without type information.
     */
    @Override
    public int compareTo(TypeReference<T> o) { return 0; }
    // just need an implementation, not a good one... hence ^^^
}

When I do this, everything is fine:
TypeReference typeReference = new TypeReference<List<Foo>> { };

This does not work, type information is lost:
class Bar<T> {
      public Bar() {
             TypeReference typeReference = new TypeReference<List<T>> { };
      }
}

Bar<Foo> br = new Bar<Foo>;

I'm aware of cleaning and all that. But if you look in the debugger, then inside the TypeReference in the first case it will be like this:
_type = "java.util.List<Foo>"
In the second:
_type = "T"
Not "Object", but "T".
Can someone explain to me what is happening in this example?

Answer the question

In order to leave comments, you need to log in

2 answer(s)
Y
YV, 2015-09-14
@guevara

Type superClass = getClass().getGenericSuperclass();

Reading the getGenericSuperclass() doc :
If the superclass is a parameterized type, the {@code Type}
object returned must accurately reflect the actual type
parameters used in the source code.

Didn't find what you were looking for?

Ask your question

Ask a Question

731 491 924 answers to any question