I am writing two extension functions for the same class:
class Something<T:Any> { ... }
They look like:
fun Something<Int>.toJson(): String = ...
fun Something<Double>.toJson(): String = ...
And results in the compiler error:
Kotlin: Platform declaration clash: The following declarations have the same JVM signature
How can I create two extension functions with only the generics signature differing? or is it not possible?
Note: this question is intentionally written and answered by the author (Self-Answered Questions), so that the answers to commonly asked Kotlin topics are present in SO. It originated in Kotlin slack #general channel.
Kotlin has the @JvmName
annotation specifically for this type of use case. In Kotlin, there isn't a problem because it knows the difference between the methods. But the Java compatible byte code would have a conflict for naming since the generics erased signatures would be identical.
Therefore you need to use this annotation to control the name from the perspective of Java and the JVM. Your Kotlin code will not see this alternative name and will use the name as you intended.
Change your code to:
@JvmName("somethingIntToJson") fun Something<Int>.toJson(): String = ...
@JvmName("somethingDoubleToJson") fun Something<Double>.toJson(): String = ...
From Kotlin, use normally:
val someIntyThing = Something<Int>(194)
val json = someIntyThing.toJson()