Tomer
04/04/2025, 2:10 PM@UnsafeVarianceContainer<out T>Optional<T>Optional<@UnsafeVariance T>Optional<out T>Optional<? extends T>@UnsafeVarianceOptionalTTOptional<T>Optionalout TDmitry Khalanskiy [JB]
04/04/2025, 2:33 PMimport java.util.Optional
class Container<out T>(val v: T?)
fun <U: Any> Container<U>.optional(): Optional<U> =
    Optional.ofNullable(v)
fun main() {
    val c = Container(3)
    println(c.optional())
}Tomer
04/04/2025, 2:34 PMDmitry Khalanskiy [JB]
04/04/2025, 2:35 PMI would have assumed they get exposed as standalone functions that take the receiver as the first parameter or somethingThat's exactly what happens, yes.
Tomer
04/04/2025, 2:36 PMTomer
04/04/2025, 2:36 PMContainer<out T>Tomer
04/04/2025, 2:37 PM@UnsafeVarianceTomer
04/04/2025, 2:37 PMDmitry Khalanskiy [JB]
04/04/2025, 2:46 PM@UnsafeVarianceOptional<T>interface MutableOptional<T> {
    fun mutate(newValue: T)
}
class Container<out T>(v: T) {
    
    private var vValue: T = v
    fun optional(): MutableOptional<@UnsafeVariance T> = object: MutableOptional<@UnsafeVariance T> {
        override fun mutate(newValue: @UnsafeVariance T) {
            vValue = newValue
        }
    }
    
    val v: T get() = vValue
}
fun main() {
    val c = Container<Int>(3)
    val c2: Container<Any?> = c
    val o: MutableOptional<Any?> = c2.optional()
    o.mutate("4.3")
    println(c.v + 10) // throws an exception
}OptionalMutableOptionalOptionaloutTomer
04/04/2025, 2:48 PMOptional<T>Dmitry Khalanskiy [JB]
04/04/2025, 2:49 PMTomer
04/04/2025, 2:49 PM