จะสร้างอินสแตนซ์ของอินเทอร์เฟซที่ไม่ระบุชื่อใน Kotlin ได้อย่างไร?


95

ฉันมีไลบรารี Java ของบุคคลที่สามซึ่งอ็อบเจ็กต์ที่มีอินเทอร์เฟซเช่นนี้:

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

ฉันจะใช้มันอย่างกระชับใน Kotlin คล้ายกับ Java anonymous class เช่นนี้ได้อย่างไร:

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

5
สำหรับทุกคนที่ดิ้นรน ฉันคิดว่าต้องประกาศอินเทอร์เฟซใน java ฉันคิดว่าการแปลง SAM ไม่ทำงานสำหรับอินเตอร์เฟส kotlin หากเป็นอินเทอร์เฟซ kotlin คุณต้องใช้ object: Handler {} way ต่อที่นี่: youtrack.jetbrains.com/issue/KT-7770
j2emanue

2
คุณสามารถทำได้ด้วยอินเทอร์เฟซ Kotlin ตั้งแต่ 1.4 - คุณเพียงแค่ประกาศว่าเป็นไฟล์fun interface.
นิค

18

ฉันมีกรณีที่ฉันไม่ต้องการสร้าง var สำหรับมัน แต่ทำแบบอินไลน์ วิธีที่ฉันประสบความสำเร็จคือ

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

                        override fun OnMethod2() {}
})


โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.