1. Lambda表达式的本质
Lambda表达式和匿名内部类,其实都是用来生成一个接口的实例。
与通过类来实现接口,然后类实例化的一个对象相比,Lambda表达式和匿名内部类语法更加简洁,不必去定义类,就可以生成接口的实例。
2. 语法
例如加密这个事情,加密接口如下:
/**
* 加密接口
*/
public interface IEncoder {
/**
* 对数字加密
*/
public int encode(int num);
}
2.1 通过类实现加密
要实现具体的加密算法,最普通的方式就是定义类来实现接口,下面这段代码纯属常规操作,无需解释。
/**
* 加法加密
*/
public class AddEncoder implements IEncoder {
@Override
public int encode(int num) {
return num + 1;
}
/**
* 测试
*/
public static void main(String[] args) {
IEncoder encoder = new AddEncoder();
int result = encoder.encode(1);
System.out.println("加密结果:" + result);
}
}
2.2 通过匿名内部类实现加密
如果通过匿名内部类的话,则无需专门定义一个类,直接在需要的时候定义一个匿名的内部类即可。
public class AnonymousDemo {
public static void main(String[] args) {
// 使用时临时定义类
IEncoder encoder = new IEncoder() {
@Override
public int encode(int num) {
return num + 1;
}
};
int result = encoder.encode(1);
System.out.println("加密结果:" + result);
}
}
对于上面两个例子中的encoder对象来说,他的能力其实是一样的,都是一个具备加法加密方法的实例。但是在具体产生这个对象的方式上,匿名内部类可以临时起意,进行加密方法的定制,相比与专门定义一个类更加轻量级。
2.3 通过Lambda表达式实现加密
Lambda表达式更加简单,但是本质上依然是生成一个接口实例。
public class LambdaDemo {
public static void main(String[] args) {
// 通过表达式定义接口的实力
IEncoder encoder = (num) -> {
return num + 1;
};
int result = encoder.encode(1);
System.out.println("加密结果:" + result);
}
}
可以看到,其实Lambda表达式,还是在表现一个方法的具体实现。
(num)表示参数
->表示这是一个Lambda表达式
{return num+1;}代码块,编写具体的方法逻辑。
2.4 Lambda表达式的简略写法
当参数列表只有一个参数时,可以省略参数的圆括号。
当代码块中只有一条语句时,可以直接省略代码块的{}。
当代码块只有一个return时,也可以省略。
所以上面的表达式可以简化为:
public class LambdaDemo {
public static void main(String[] args) {
// 通过表达式定义接口的实力
IEncoder encoder = num -> num + 1;
int result = encoder.encode(1);
System.out.println("加密结果:" + result);
}
}
3. Lambda表达式存在的意义
匿名内部类和Lambda表示式其实都是为了简化接口实例化的代码。
当接口的实例仅仅是拿过来使用1次时,单独去写一个类有点浪费了,不如直接一个表达式搞定。
Lambda比匿名内部类更加简单,但是Lambda表达式只能面向只有一个方法的接口,而匿名函数可以支持有任意多个方法的接口
4. 函数式接口
Lambda表达式能够支持的,只有一个方法的接口,可以被称为函数式接口。
Java8为函数式接口提供了一个注解@FunctionalInterface,用来检查接口是否为函数式接口,可见函数式接口的重要性!
语法如下:
/**
* 加密接口
*/
@FunctionalInterface
public interface IEncoder {
/**
* 对数字加密
*/
public int encode(int num);
}