Kotlin 1.3

What's new?

Kittinun Vantasin

About me

Mobile Enthusiastic 📱🍭

💖 Kotlin + Open source

Works at Merpay in Tokyo 🗼

github.com/kittinunf

@kittinunf

About me

from wonderful land of smiles ☺️

loves thai food because it is yummy 😋

rides Tuktuk while in Bangkok 💨

Kotlin 1.3

2010  Started

2011  Revealed

Late 2012  OSS

2014 

Gain popularity

2016

Hit 1.0

2017

Hit 1.1

Late 2017

Hit 1.2

2018

Hit 1.3

What's new?

Contracts

Capture variable in local when scope

@JvmStatic & @JvmField in interfaces

fun main()

255 arity function

inline classes! + USignedTypes

consistent empty related functions

etc.

Guide

  
  Kotlin 1.2.71 or earlier
  
  Awesome Kotlin 1.3

Contracts

  
  fun foo(s: String?) {
    s?.let {
      s.length // smart cast is here
    }
    
    if (s.isNotNull()) {
      s?.length // no smart cast here :(
    }
  }

  fun String?.isNotNull() = this != null
  
  fun foo(s: String?) {
    s?.let {
      s.length // smart cast here
    }

    if (isNotNull(s)) {
      s.length // also smart cast here
    }
  }

  fun isNotNull(value: Any?): Boolean {
    contract { returns(true) implies (value != null) }
    return value != null
  }

Contracts cont.

  
  
  val myValue: String
  
  myRun {
    myValue = "Hello, World!"   // Error here ... :(
  }
  
  fun <R> myRun(block: () -> R): R {
    return block()
  }
  
  
  val myValue: String
  
  myRun {
    myValue = "Hello, World!"   // Woohoo it works!
  }
  
  fun <R> myRun(block: () -> R): R {
    contract {
      callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    return block()
  }

Capture when scope

  
  val answer = calculate()
  
  return when(answer) {
    42 -> "Right answer!"
    123 -> "ARGHHHHH!"
    else -> "Sorry, $answer is not correct"
  }

  answer // I don't want to see you here

  
  

  return when(val answer = calculate()) {
    42 -> "Right answer!"
    123 -> "ARGHHHHH!"
    else -> "Sorry, $answer is not correct"
  }


  //answer you are not being seen here

@JvmStatic & @JvmField 

  
interface Foo {
    
  companion object {
        
    @JvmField
    val bar: Int = 42
        
    @JvmStatic
    fun sayHello(name: String = "") {
      println("Hello, $name")
    }
  }
}
  

fun main() 

  
  fun main() {
    println("Hello, world")
  }


  // compare to the full-blown version

  fun main(args : Array<String>) { 
    println("Hello, world") 
  }

  

255 Arity function 

  
  
  fun onlyRealManCanUnderstandThis(
     block: (Any, Any, Any, /* 250 more */, Any, Any
  ) {
    block(Any(), Any(), /* 250 more */, Any(), Any())
  }
  
  
  @SinceKotlin("1.3")
  interface FunctionN<out R> : Function<R> {
    operator fun invoke(vararg args: Any?): R
    
    override val arity: Int
  }

Inline classes!!

  
  
  class FirstNameInKana(n: String) 
  class LastNameInKana(n: String)

  fun formattedName(f: FirstName, l: LastName) {
    println("$l $f")
  }

  formattedName(
    FirstNameInKana("やまだ"), 
    LastNameInKana("たろう")
  )

  typealias FirstName = String // OK, but meh :|
  typealias LastName = String // OK, but meh :|
  

  inline class FirstNameInKana(n: String) 
  inline class LastNameInKana(n: String)

  fun formattedName(f: FirstName, l: LastName) {
    println("$l $f")
  }

  formattedName(
    FirstNameInKana("やまだ"), 
    LastNameInKana("たろう")
  )
  
  

  
  

Inline classes cont.

Must have a public constructor with a single val

No init block

Must be final

Cannot extend anything - implement interfaces OK

Cannot have any additional properties

Must be a top level class

HANDLE WITH CARE

UTypes FTW

  
  
  // Literal shortcode for UInt
  val aUInt: UInt = 3u
  val aULong: ULong = 3uL

  // And from Hex...
  val hexUByte: UByte = 0xFFu  // 255
  val hexUShort: UShort = 0xFFFFu  // 65535
  val hexUInt: UInt = 0xFFFF_FFFFu  // 4294967295
  val hexULong: ULong = 0xFFFF_FFFF_FFFF_FFFFu  // xD

Empty related func.

  
  
  fun calculateArray(): Array<Int>? {
    // some calculation

    
  }

  val array = calculateArray()

  array.isNullOrEmpty() // ERRRAARGGHHH, there is none
  
  
  fun calculateArray(): Array<Int>? {
    // heavy calculation

    
  }

  val array = calculateArray()

  if (array.isNullOrEmpty()) {
    // :) Joy
  }
  
  
  val maybeArray: Array<T>
  maybeArray.isNullOrEmpty()


  val maybeCollection: Collection<T>
  maybeCollection.isNullOrEmpty()


  val maybeMap: Map<T, U>
  maybeMap.isNullOrEmpty()
  
  
  

Empty related func.

  
  
  val listOfFeatures: List<String> = ...
  println(listOfFeatures.ifEmpty { 
     listOf("Default1", "Default2") 
  })

  
  val townProvidedByUser: String = ...
  println(townProvidedByUser.ifBlank { "東京" })

Serialization Library

  
  @Serializable
  data class Destination(
        val name: String? = "",
        val country : String? = "",
        val code : Int = 0,
        @Optional
        val isMetro : Boolean = false,
        @Transient
        val favorite : Boolean = false
  )
  
  val tokyoAsString = JSON.stringify(
    Destination(name = "東京", country = "Japan") 
  ) // { "name":"東京","country":"Japan","code":0 }
  

  val tokyo = JSON.unquoted.parse<Destination>(
    "{ name:東京,country:Japan, code:12 }"
  ) // Destination(name = "東京", country = "Japan", code = 12)

Caveats

The experimental feature is really experimenting!

They can be breaking for the next version! :)

How do you feel about 1.3?

Q & A

Kotlin 1.3, What's new?

By Kittinun Vantasin

Kotlin 1.3, What's new?

  • 1,224