Wie erstelle ich eine Instanz einer anonymen Schnittstelle in Kotlin?

89

Ich habe eine Java-Bibliothek eines Drittanbieters, die ein Objekt mit einer solchen Schnittstelle hat:

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

Wie kann ich es in Kotlin ähnlich wie in einer anonymen Java-Klasse wie folgt präzise implementieren:

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
quelle

Antworten:

130

Angenommen, die Schnittstelle verfügt nur über eine einzige Methode, können Sie SAM verwenden

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

Wenn Sie eine Methode haben, die einen Handler akzeptiert, können Sie sogar Typargumente weglassen:

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

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

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

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

Wenn die Schnittstelle mehr als eine Methode hat, ist die Syntax etwas ausführlicher:

val handler = object: Handler2<String> {
    override fun call(context: String?) { println("Call: $context") }
    override fun run(context: String?) { println("Run: $context")  }
}
Miensol
quelle
2
acceptHandler { println("Hello: $it")}würde auch in den meisten Fällen
funktionieren
5
Für alle, die Probleme haben. Ich denke, die Schnittstelle muss in Java deklariert werden. Ich denke, SAM-Konvertierung funktioniert nicht für Kotlin-Schnittstellen. Wenn es sich um eine Kotlin-Schnittstelle handelt, müssen Sie das Objekt verwenden: Handler {} way. per hier: youtrack.jetbrains.com/issue/KT-7770 .
j2emanue
14

Ich hatte einen Fall, in dem ich keine Variable dafür erstellen wollte, sondern sie inline ausführen wollte. So wie ich es erreicht habe

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

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

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

Die einfachste Antwort ist wahrscheinlich das Lambda des Kotlin:

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

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