Jak stworzyć instancję anonimowego interfejsu w Kotlinie?

97

Mam bibliotekę Java innej firmy, która jest obiektem z takim interfejsem:

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

Jak mogę zwięźle zaimplementować to w Kotlin, podobnie jak anonimowa klasa Java, taka jak ta:

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
źródło

Odpowiedzi:

146

Zakładając, że interfejs ma tylko jedną metodę, możesz skorzystać z SAM

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

Jeśli masz metodę, która akceptuje procedurę obsługi, możesz nawet pominąć argumenty typu:

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

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

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

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

Jeśli interfejs ma więcej niż jedną metodę, składnia jest nieco bardziej szczegółowa:

val handler = object: Handler2<String> {
    override fun call(context: String?) { println("Call: $context") }
    override fun run(context: String?) { println("Run: $context")  }
}
miensol
źródło
2
acceptHandler { println("Hello: $it")}działałby również w większości przypadków
voddan
5
Dla każdego walczącego. Myślę, że interfejs musi być zadeklarowany w java. Myślę, że konwersja SAM nie działa dla interfejsów kotlin. jeśli jest to interfejs kotlin, musisz użyć obiektu: Handler {} w sposób. tutaj: youtrack.jetbrains.com/issue/KT-7770 .
j2emanue
2
Możesz to zrobić za pomocą interfejsu Kotlin od 1.4 - po prostu deklarujesz go jako plik fun interface.
Nick
18

Miałem przypadek, w którym nie chciałem tworzyć dla niego zmiennej, ale robiłem to w tekście. Sposób, w jaki to osiągnąłem, jest

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

                        override fun OnMethod2() {}
})
Aalap
źródło
15
     val obj = object : MyInterface {
         override fun function1(arg:Int) { ... }

         override fun function12(arg:Int,arg:Int) { ... }
     }
pruthwiraj.kadam
źródło
2

Najprostszą odpowiedzią jest prawdopodobnie lambda Kotlina:

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

handler.call(myContext) // Prints "Hello world"
Peter Lamberg
źródło