Como criar uma instância de interface anônima em Kotlin?

97

Tenho uma biblioteca Java de terceiros que possui um objeto com interface como esta:

public interface Handler<C> {
  void call(C context) throws Exception;
}

Como posso implementá-lo de forma concisa em Kotlin semelhante à classe anônima Java desta forma:

Handler<MyContext> handler = new Handler<MyContext> {
   @Override
   public void call(MyContext context) throws Exception {
      System.out.println("Hello world");
   }
}

handler.call(myContext) // Prints "Hello world"
Peter Lamberg
fonte

Respostas:

146

Supondo que a interface tenha apenas um único método, você pode usar o SAM

val handler = Handler<String> { println("Hello: $it") }

Se você tem um método que aceita um manipulador, pode até omitir os argumentos de tipo:

fun acceptHandler(handler:Handler<String>){}

acceptHandler(Handler { println("Hello: $it") })

acceptHandler({ println("Hello: $it") })

acceptHandler { println("Hello: $it") }

Se a interface tiver mais de um método, a sintaxe será um pouco mais detalhada:

val handler = object: Handler2<String> {
    override fun call(context: String?) { println("Call: $context") }
    override fun run(context: String?) { println("Run: $context")  }
}
miensol
fonte
2
acceptHandler { println("Hello: $it")}também funcionaria na maioria dos casos
voddan
5
Para quem está lutando. Acho que a interface deve ser declarada em java. Eu acho que a conversão de SAM não está funcionando para interfaces kotlin. se for uma interface kotlin, você deve usar o objeto: Maneira {} do manipulador. por aqui: youtrack.jetbrains.com/issue/KT-7770 .
j2emanue
2
Você pode fazer isso com uma interface Kotlin a partir de 1.4 - basta declará-la como um fun interface.
Nick
18

Tive um caso em que não queria criar uma var para ele, mas fazê-lo embutido. A forma como consegui isso é

funA(object: InterfaceListener {
                        override fun OnMethod1() {}

                        override fun OnMethod2() {}
})
Aalap
fonte
15
     val obj = object : MyInterface {
         override fun function1(arg:Int) { ... }

         override fun function12(arg:Int,arg:Int) { ... }
     }
pruthwiraj.kadam
fonte
2

A resposta mais simples provavelmente é o lambda de Kotlin:

val handler = Handler<MyContext> {
  println("Hello world")
}

handler.call(myContext) // Prints "Hello world"
Peter Lamberg
fonte