提到lambda表达式,就不得不提一下函数式表达式。函数式表达式与面向对象编程的最大区别是
面向对象编程是对数据进行抽象,而函数式编程是对行为进行抽象。
函数式编程是能更直接的表达程序员的业务逻辑意图,让程序员不必纠缠匿名内部类的冗繁和可读性,让事件处理系统也变得更简单,编写惰性代码也更容易(惰性代码即在真正需要的时候才初始化的变量值)。
牢记函数式编程的核心思想:使用不可变值和函数,函数对一个值进行处理,映射成另一个值。
第一个lambda表达式
先来看看android里的常用的匿名内部类的button点击事件写法:
btnTest.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Toast.makeText(MainActivity.this,"点击了",Toast.LENGTH_SHORT);
}
});
lambda
btnTest.setOnClickListener(view -> Toast.makeText(MainActivity.this,"点击了",Toast.LENGTH_SHORT));
在这里我们传入了一段代码块,一个没名字的函数,view是参数和上面的内部类是同一个参数, -> 将参数和lambda主体分开。lambda无需显式指定参数类型,这是因为jac会在编译时候会根据上下文进行类型推导(注:有事为了提升代码可读性,根据习惯仍需指定类型,编译器不一定能根据上下文推导出类型)
lambda的其它表现形式
//不包含参数()表示,返回类型为void
Runnable mRunnable = () -> System.out.print("hello");
//表示lambda也可以是代码块,与普通代码块并无区别,可返回或抛异常
Runnable mRunnable1 = () -> {
System.out.print("hello");
System.out.print("world");
};
//有且只有一个参数,可省略参数括号
View.OnClickListener onClickListener = view -> Toast.makeText(MainActivity.this, "点击了", Toast.LENGTH_SHORT);
//表示包含多个参数,此代码意思并不是表示是求和,而是创建一个函数用来计算两数相加的结果,
//变量add类型是 BinaryOperator<Long> ,它的值不是x+y的和,而是x+y这句代码。
// BinaryOperator<Long> add = (x, y) -> x + y;
//同上,不同的是参数显式的申明了参数类型
BinaryOperator<Long> add = (Long x, Long y) -> x + y;
注:目标类型是指lambda表达式上下文的环境类型,比如将lambda表达式赋值给局部变量,或传递给一个方法作为参数,局部变量或方法参数类型就是lambda表达式的类型。
匿名内部类使用方法里的局部变量必须申明为final,
//此时namefinal为隐式,如果改变name值属性就好报错
(final) String name ="cb";
//在此改变name值编译器不通过。
name = "bc";
btnTest = (Button) findViewById(R.id.btn_test);
btnTest.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Toast.makeText(MainActivity.this, "点击了"+name, Toast.LENGTH_SHORT);
}
});
btnTest = (Button) findViewById(R.id.btn_test);
btnTest.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Toast.makeText(MainActivity.this, "点击了"+name, Toast.LENGTH_SHORT);
}
});
btnTest.setOnClickListener(view -> Toast.makeText(MainActivity.this, "点击了"+name, Toast.LENGTH_SHORT));
即lambda表达式引用的是值,而不是变量。同时也解释了lambda表达式为什么是闭包,未赋值的变量与周边环境隔离起来,进而被绑定到一个特定的值里。