Lambda表达式可以很好的去定义一个 匿名函数,避免我们去写一些某些函数的抽象类和接口,然后去实现它们,在kotlin中我们 可以将一个函数作为另一个函数的参数
简化setOnclickListener ()
Android中的常用额点击事件 View.setOnclickListenr(),如果使用java中的代码去编写 ,需要下去书写一个接口,
public interface OnClickListener {
void onClick(View v);
}
然后编写一个内部类 去实现
view.setOnClickListener(new OnClickListener(){
@Override
public void onClick(View v) {
Toast.makeText(v.getContext(), "Click", Toast.LENGTH_SHO
RT).show();
}
})
我们将把上面的代码转换成Kotlin(使用了Anko的toast函数):
view.setOnClickListener(object : View.OnClickListener {
override fun onClick(v: View) {
toast("Click")
}
})
kotlin中对java 库做了一些优化.
fun setOnClickListener(listener: (View) -> Unit) {
}
在这个例子中,我们接收一个View,然后返回一个Unit( 没有东西)。所以根据这种思想,我们可以把前面的代码简化成这样:
view.setOnClickListener({ view -> toast("Click")})
如果函数的最后一个参数是个函数,我们可以省略这个括号
view.setOnClickListener { toast("Click") }
with
由于这些的改变我们已经使用一些代码块,和有趣的函数比如说
inline fun <T> with(t: T, body: T.() -> Unit) { t.body() }
参数接受一个 T 类型的对象和一个作为扩展函数的函数,它的实现仅仅是让这个对象来来执行这个函数,因为第二个参数是个 函数 我们可以把它放在括号外面,创建一个代码块,用this和访问所有的 public的方法和属性
with(forecast) {
Picasso.with(itemView.ctx).load(iconUrl).into(iconView)
dateView.text = date
descriptionView.text = description
maxTemperatureView.text = "$high"
minTemperatureView.text = "$low"
itemView.setOnClickListener { itemClick(this) }
}
内联函数和普通函数的区别 一个内联函数被在编译的时候替换掉而不是真正的方法调用,这样可以减少内存的分配和运行时的开销;如果是个普通函数那么他会创建一个函数的额对象
inline fun supportsLollipop(code: () -> Unit) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
code()
}
}
检查版本完成后可以去执行如下的代码:
supportsLollipop {
window.setStatusBarColor(Color.BLACK)
}
let
函数定义:
public inline fun <T, R> T.let(block: (T) -> R): R = block(this)
调用某对象的let函数,则该对象是函数的参数,在函数块内可通过 it 来指代该对象,返回值为 函数块的最后一行或者是指定的 return 的表达式
val a = "string".let {
println(it)
3
}
println(a)
apply
public inline fun <T> T.apply(block: T.() -> Unit): T { block(); return this }
调用某对象的apply函数,在函数块内可以通过 this 指代该对象。返回值为该对象自己。
val a = "string".apply {
println(this)
}
println(a)