Як створити екземпляр анонімного інтерфейсу в Kotlin?


95

У мене є стороння бібліотека Java, яка є об’єктом з таким інтерфейсом:

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

Як я можу коротко реалізувати його в Kotlin, подібному до анонімного класу Java, як це:

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"

Відповіді:


143

Якщо припустити, що інтерфейс має лише один метод, ви можете використовувати SAM

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

Якщо у вас є метод, який приймає обробник, ви можете навіть пропустити аргументи типу:

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

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

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

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

Якщо інтерфейс має більше одного методу, синтаксис трохи детальніший:

val handler = object: Handler2<String> {
    override fun call(context: String?) { println("Call: $context") }
    override fun run(context: String?) { println("Run: $context")  }
}

2
acceptHandler { println("Hello: $it")}також працювало б у більшості випадків
воддан

5
Для тих, хто бореться. Я думаю, що інтерфейс повинен бути оголошений у Java. я думаю, що перетворення SAM не працює для інтерфейсів kotlin. якщо це інтерфейс kotlin, вам доведеться використовувати об'єкт: Handler {} шлях. тут: youtrack.jetbrains.com/issue/KT-7770 .
j2emanue

2
Ви можете зробити це за допомогою інтерфейсу Kotlin станом на 1.4 - ви просто оголосите це як fun interface.
Нік

18

У мене був випадок, коли я не хотів створювати для нього var, а робити це вбудовано. Я досяг цього

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

                        override fun OnMethod2() {}
})

14
     val obj = object : MyInterface {
         override fun function1(arg:Int) { ... }

         override fun function12(arg:Int,arg:Int) { ... }
     }

2

Ймовірно, найпростіша відповідь - лямбда Котліна:

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

handler.call(myContext) // Prints "Hello world"
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.