35
votes

In this simplified example I have a generic class, and a method that returns a Map regardless of the type parameter. Why does the compiler wipe out the types on the map when I don't specify a type on the containing class?

import java.util.Map;

public class MyClass<T>
{
    public Map<String, String> getMap()
    {   
        return null;
    }

    public void test()
    {   
        MyClass<Object> success = new MyClass<Object>();
        String s = success.getMap().get("");

        MyClass unchecked = new MyClass();
        Map<String, String> map = unchecked.getMap();  // Unchecked warning, why?
        String s2 = map.get("");

        MyClass fail = new MyClass();
        String s3 = fail.getMap().get("");  // Compiler error, why?
    }
}

I get this compiler error.

MyClass.java:20: incompatible types
found   : java.lang.Object
required: java.lang.String
                String s3 = fail.getMap().get("");  // Compiler error
4
What is the exact text of the unchecked warning?Powerlord
warning: [unchecked] unchecked conversionCraig P. Motlin

4 Answers

37
votes

Got it. This actually isn't a bug, strange as it might seem.

From section 4.8 (raw types) of the JLS:

The type of a constructor (§8.8), instance method (§8.8, §9.4), or non-static field (§8.3) M of a raw type C that is not inherited from its superclasses or superinterfaces is the erasure of its type in the generic declaration corresponding to C. The type of a static member of a raw type C is the same as its type in the generic declaration corresponding to C.

So even though the method's type signature doesn't use any type parameters of the class itself, type erasure kicks in and the signature becomes effectively

public Map getMap()

In other words, I think you can imagine a raw type as being the same API as the generic type but with all <X> bits removed from everywhere (in the API, not the implementation).

EDIT: This code:

MyClass unchecked = new MyClass();
Map<String, String> map = unchecked.getMap();  // Unchecked warning, why?
String s2 = map.get("");

compiles because there's an implicit but unchecked conversion from the raw Map type to Map<String, String>. You can get the same effect by making an explicit conversion (which does nothing at execution time) in the last case:

// Compiles, but with an unchecked warning
String x = ((Map<String, String>)fail.getMap()).get("");
5
votes

Hm ... unfortunately I can't tell you why it fails. But I can give you a simple workaround:

Change the type of fail to MyClass<?>, then it will compile just fine.

3
votes

Very interesting question, and very interesting answer by Jon Skeet.

I just want to add something about the stupidity or not stupidity of this behaviour of the java compiler.

I think that the compiler assumes that if you don't specify the type parameter in a generc class you are not able (or don't want to) use any type parameter at all. You could use a version of java earlier than 5, or love to make casts manually.

It doesn't seem so stupid to me.

1
votes

Generic types get erased after compiling.

When you do:

Map<String, String> map = unchecked.getMap();

you're forcing a cast from Map to Map<String, String>, and that's why the unchecked warning. However, after that you can do:

String s2 = map.get("");

because map is of type Map<String, String>.

However, when you do

String s3 = fail.getMap().get(""); 

you're not casting fail.getMap() to anything, so it's considered to be plainly Map, not Map<String, String>.

What you should do in the latter is something like:

String s3 = ((Map<String, String>fail.getMap()).get("");

which will still thrown a warning but will work anyway.