阅读完需:约 2 分钟
只有一个抽象方法的接口称为函数式接口或 SAM(单一抽象方法)接口。函数式接口可以有多个非抽象成员,但只能有一个抽象成员。
可以用 fun
修饰符在 Kotlin 中声明一个函数式接口。
fun interface KRunnable {
fun invoke()
}
java—SAM

在java中对于SAM接口的调用一般都是用java的内部匿名类来实现的。

但是在java8中有lambda表达式可以调用,但是java8的lambda表达式和kotlin的lambda表达式是不一样的,因为kotlin的lambda表达式是函数,在kotlin中是一等公民的存在。

java的SAM其实上就是函数匿名类的一种转换。
Kotlin—SAM
对于函数式接口,可以通过 lambda 表达式实现 SAM 转换,从而使代码更简洁、更有可读性。
使用 lambda 表达式可以替代手动创建实现函数式接口的类。 通过 SAM 转换, Kotlin 可以将其签名与接口的单个抽象方法的签名匹配的任何 lambda 表达式转换为实现该接口的类的实例。

例如,有这样一个 Kotlin 函数式接口:
fun interface IntPredicate {
fun accept(i: Int): Boolean
}
如果不使用 SAM 转换,那么你需要像这样编写代码:
// 创建一个类的实例
val isEven = object : IntPredicate {
override fun accept(i: Int): Boolean {
return i % 2 == 0
}
}
通过利用 Kotlin 的 SAM 转换,可以改为以下等效代码:
// 通过 lambda 表达式创建一个实例
val isEven = IntPredicate { it % 2 == 0 }
可以通过更短的 lambda 表达式替换所有不必要的代码。
fun interface IntPredicate {
fun accept(i: Int): Boolean
}
val isEven = IntPredicate { it % 2 == 0 }
fun main() {
println("Is 7 even? - ${isEven.accept(7)}")
}
返回:
Is 7 even? - false
SAM演变示意



