At the bytecode level the method will have a descriptor that just says, there's a method with the name getHashMap, that takes no arguments and returns a Map (no generics).
Then when the compiler is analyzing the line Map<String, Map<Long, List<String>>> map = getHashMap();
, and it will say, ok, I need to have a variable with a declared type of Map<String, Map<Long, List<String>>>
, but to actually get the instance I need to call a method. At this point it's the compiler's job to check if the return type of the method matches the declared type of the variable your assigning that result to. So it checks if String
matches K
, and if Map<Long, List<String>>
matches V, which they do, so it considers that the assignment is type safe, and generates the bytecode which basically uses a Map (no generics) variable.
If you would have declared your method as:
static <K extends Number,V> Map<K,V> getHashMap()
{
return new HashMap<K, V>();
}
when analyzing the assignment, the compiler would see that String
doesn't match K extends Number
, would throw a compilation error, and won't create bytecode for that assignment.