Lambda简介
Lambda表达式是Java8引入的一个重要特性,相当于一个语法糖。
语法糖(Syntactic sugar)是指在编程语言中引入的一种语法,它可以使代码更易读、更简洁,但并没有引入新的功能或改变语言的底层机制。语法糖并不会改变语言的语义,只是提供了一种更方便的编写方式。
- Lambda表达式可以被视为匿名函数
- 允许在需要函数的地方以更简洁的方式定义功能
使用条件:只要是函数式接口就可以用Lambda表达式简化
函数式接口:接口中有且只有一个未实现的方法,这个接口就叫函数式接口
如果接口中有超过一个未实现方法,则不是函数式接口,不能用Lambda表达式 如果接口中有一个未实现方法,有一个默认实现方法,则是函数式接口,可以用Lambda表达式
如:
// 接口中有超过一个未实现方法,不是函数式接口 interface MyInterface { int sum(int a, int b); int min(int a, int b); } // 接口中只有一个未实现的方法,是函数式接口 interface MyCase{ int hello(); default int hello(int a){return a;} //默认实现 }
可以用jdk中提供的检查注解@FunctionalInterface来检查该接口是否为函数式接口
正常情况:
异常情况:
Lambda表达式与函数式接口
举个例子:
有一个接口MyInterface,里面有一个方法sum()。
interface MyInterface { int sum(int a, int b); }
如果想要实现这个接口,我们可以使用两种方法。
1.自己写实现类
class MyInterfaceImpl implements MyInterface { @Override public int sum(int a, int b) { return a + b; } }
然后调用
public class Lambda { public static void main(String[] args) { MyInterface myInterface = new MyInterfaceImpl(); int result = myInterface.sum(1, 2); System.out.println(result); } }
2.创建匿名实现类如果每个接口都要写实现类的话,总觉得会有点麻烦。而且,如果在以后的业务中,要实现的方法不是两数之和,而是两数的平方和的话,那我岂不是要再写一个实现类?为了解决这个问题,我们可以采用匿名实现类,动态的去实现接口。
public class Lambda { public static void main(String[] args) { // 1. 自己创建实现类对象 MyInterface myInterface = new MyInterfaceImpl(); int result = myInterface.sum(1, 2); System.out.println("我是 自己创建的实现类对象 " + result); // 2. 创建匿名实现类 MyInterface myInterface1 = new MyInterface() { @Override public int sum(int a, int b) { return a*a + b*b; } }; int result1 = myInterface1.sum(1, 2); System.out.println("我是 匿名实现类 " + result1); } }
运行结果:
可以看到,每次创建匿名实现类的时候,有很多格式上的东西是每次都要写的,这样就很冗余。比如下面我选中的这部分:
这部分在上面的接口interface MyInterface中就已经定死了。就算不写new MyInterface(),从创建匿名实现类的前面部分MyInterface myInterface1,也可以看出实现的就是MyInterface接口。
与前面相比,lambda表达式只保留动态的东西,把写死的东西去掉。 用lambda表达式的方式实现接口:
// 3. Lambda表达式实现接口 参数列表 + 箭头 + 方法体 MyInterface myInterface2 = (int a, int b) -> { return a * a + b * b; };
这是lambda表达式的完整写法,然而我们可以看到,入参的类型在接口中也是定好的。这就说明还有更简化的写法:
1.参数类型可以不写,只写参数名,参数变量名随意定义
MyInterface myInterface3 = (x, y) -> { return x * x + y * y; };
参数名不一定是接口中定义的(a,b),也可以定义为其他的名字,比如(x,y)
2.参数部分在没有入参的时候,最少可以只有一个(),但是不能不写括号!!!!或者在只有一个入参的时候,只有一个参数名
interface MyCase{ int hello(); } public class Lambda { public static void main(String[] args) { MyCase myCase = () -> { return 1; }; } } --------------------------------------------------- interface MyCase1{ int hello(int a); } public class Lambda { public static void main(String[] args) { MyCase1 myCase1 = a -> { return a + 1; }; } }
3.方法体只有一句话的时候,{}和return可以省略
interface MyCase1{ int hello(int a); } public class Lambda { public static void main(String[] args) { MyCase1 myCase11 = a -> a + 2; // 调用方法 System.out.println(myCase11.hello(1)); } }
Lambda表达式使用
未来使用函数时接口会比较频繁,当调用某个方法传入参数,这个参数实例是一个接口对象,且只定义了一个方法,就可以直接用Lambda简化写法
比较器的使用
当我们使用比较器的时候,可以使用Lambda表达式简化写法
public class Lambda { public static void main(String[] args) { ArrayList<String> names = new ArrayList<String>(); names.add("Pidanxia"); names.add("Lucy"); names.add("Bob"); names.add("Tom"); // 普通写法 Collections.sort(names,new Comparator<String>() { @Override public int compare(String o1, String o2) { return o1.compareTo(o2); } }); // Lambda写法 Collections.sort(names,(o1,o2)->o1.compareTo(o2)); } }
可以看到Lambda表达式与普通写法相比,简洁了非常多。 除了Lambda表达式之外,还可以用方法引用来简化写法。如:
Collections.sort(names, String::compareTo);
类::方法:表示引用类中的实例方法。
比如例子中的语句,就是说,以names为入参,调用String类中的compareTo方法
线程的使用
// 普通写法 new Thread(){ @Override public void run() { System.out.println("Hello"); } }.start(); // Lambda写法 new Thread(()->System.out.println("Hello")).start();标签:10,int,接口,MyInterface,搞懂,public,表达式,Lambda From: https://www.cnblogs.com/qimoxuan/p/18309421