tags:

views:

562

answers:

4

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
+2  A: 

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.

Joachim Sauer
A cast works just fine too. Why would a type of ? affect the type of the map? It still makes no sense to me.
Craig P. Motlin
@Motlin: Jon explained it pretty well: MyClass<?> is not a raw type.
Joachim Sauer
Now that I understand why your workaround works, I'm going to use it instead of a cast. Thanks.
Craig P. Motlin
+9  A: 

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("");
Jon Skeet
That's stupid. Thanks Jon!
Craig P. Motlin
Hm ... it kind of makes sense ... but it's incredibly stupid at the same time ...
Joachim Sauer
Not so stupid in my opinion.
alexmeia
+2  A: 

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.

Seb
Then why does the first one compile? "Erasure" isn't the full answer here.
Craig P. Motlin
Yup, you're right. I guess Jon's answer's the best one here.
Seb
A: 

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.

alexmeia
Why -1? It's not a polemic question, I am not so expert with back-end development, so I know that i can say something incorrect. I would like to know what's wrong in what I said.
alexmeia