Как создать экземпляр анонимного интерфейса в Котлине?

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")}также будет работать в большинстве случаев
voddan 07
5
Для всех, кто борется. Я думаю, что интерфейс должен быть объявлен в java. Я думаю, что преобразование SAM не работает для интерфейсов kotlin. если это интерфейс kotlin, вы должны использовать метод object: 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) { ... }
     }
pruthwiraj.kadam
источник
2

Самый простой ответ - это лямбда Котлина:

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

handler.call(myContext) // Prints "Hello world"
Питер Ламберг
источник