Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Does java.util.HashMap not implement java.util.Map in Kotlin?

I have trouble understanding why a Java HashMap is incompatible with a Java Map in Kotlin:

val map : java.util.Map<Int, Int> = java.util.HashMap<Int, Int>()
                                    // ERROR: Type mismatch

Is this a bug or is it intentionally enforced to be an error in Kotlin?


Here is a second example with the Java to Kotlin compiler. Take a look at this Java example file:

public class Test {
    public static void main(String[] args) {
        java.util.Map<Integer, Integer> map = new java.util.HashMap<>();
        insertValue(map);
    }

    private static void insertValue(java.util.Map<Integer, Integer> map) {
        map.putIfAbsent(0, 1);
    }
}

Running "Convert Java to Kotlin" results in this file:

object Test {
    @JvmStatic fun main(args: Array<String>) {
        val map = java.util.HashMap<Int, Int>()
        insertValue(map)
    }

    private fun insertValue(map: Map<Int, Int>) {
        map.putIfAbsent(0, 1)  // ERROR! Unresolved reference
    }
}

If I try to modify it to reflect more the original file, where insertValue expected a Java java.util.Map, I get another error:

object Test {
    @JvmStatic fun main(args: Array<String>) {
        val map = java.util.HashMap<Int, Int>()
        insertValue(map)  // ERROR: Type mismatch
    }

    private fun insertValue(map: java.util.Map<Int, Int>) {
        map.putIfAbsent(0, 1)  // is now OK
    }
}
like image 504
Philipp Claßen Avatar asked Dec 13 '15 19:12

Philipp Claßen


People also ask

What is difference between HashMap and Map in Kotlin?

Key Differences between Map and HashMapThe Map is an interface, and HashMap is a class of the Java collection framework. The Map interface can be implemented by using its implementing classes. In comparison, the HashMap class implements the Map interface. The Map contains unique key-pair values.

Is HashMap in Java Util?

HashMap<K, V> is a part of Java's collection since Java 1.2. This class is found in java. util package. It provides the basic implementation of the Map interface of Java.


2 Answers

In Kotlin, a java.util.HashMap, at compile time, does not implement java.util.Map, but implements kotlin.collections.MutableMap which extends the read-only kotlin.collections.Map.

This is because of Kotlin's mapped types. See Collections in Kotlin M3 is Out! for more details.

MutableMap does not define any putIfAbsent, but it has an extension function getOrPut.

like image 184
mfulton26 Avatar answered Oct 19 '22 13:10

mfulton26


I created a bug report, primarily for the "Java to Kotlin" converter that generates invalid Kotlin code in my original example:

KT-10400 Java to Kotlin converter: converting Java Code where a java.util.HashMap is accessed through the java.util.Map interface fails

I also mentioned that java.util.HashMap does not implement java.util.Map but I leave it to the Kotlin developers to decide whether it is a bug or whether the compiler is right to reject it.

UPDATE: The bug is marked as solved (commit 5e231db6).

like image 39
Philipp Claßen Avatar answered Oct 19 '22 14:10

Philipp Claßen