Lambda表达式的心得
如题,因为博主也是最近才接触到Lambda表达式的(PS 在这里汗颜一会)。我并不会讲解它的原理,诚然任何一件事物如果理解原理的话,使用它必将更加容易。但博主在学习的时候,大多数时候都是学会怎么用,然后在细究原理。就像你骑自行车之前,难道首先还要研究自行车的原理么?
首先Lambda表达式的最简单应用如下
Lambda表达式法
String lam= "初次相识Lambda"; new Thread(() -> System.out.println(lam)).start();
传统方法
String tradition="传统方法"; new Thread(new Runnable() { @Override public void run() { System.out.println(tradition); } }).start();
输出结果
很简洁有没有?省略了好多代码是不是,在这里 你可以发现”,Lambda表达式和在Thread创建一个匿名类的作用是一样。我们可以这样认为Lambda表达式本身代表了一个匿名类。
这就是Lambda最大的作用,当然Lambda表达式只能创建接口interface对象。 创建类是不行的,抽象类也是不行的 ,只要是类都是不行的。
首先,我定义了一个自定义的接口,可以用来测试
@FunctionalInterface public interface Lam { //Lambda表达式调用的方法 void bda(); //接口的默认方法 default void test(){ System.out.println("我是默认的方法"); }; //接口的静态方法 static void test1(){ System.out.println("我是静态方法"); } }
使用Lambda表达式 你首先要知道的
1.Lambda表达式只能是接口 interface的创建(PS从上面的例子可以看出来,Runnable是接口,可以查看源代码),并且这个接口只能包含一个方法(除了default方法和static方法)。在接口中创建default方法和static方法都必须要实现方法体如下图
2.如果你用Lambda表达式来创建类 class,则会出现以下错误 ”Target type of a lambda conversion must be an interface“如果你怕自己的定义的接口不符合Lambda表达式的规范 ,你可以在接口interfaca 上面添加注解@FunctionalInterface
3.Lambda表达式的规范表示格式 (parameters) ->{ statements; }。在某些时刻,你还可以简化这个格式
//接口定义的方法无参数时候,并且你想要执行操作也只有一句代码的时候,Lambda会自动返回一句代码,并且可以不用加{} Lam lam1=()->System.out.println("无参数");
你可以发现后面的大括号{ }没了,这是因为后面代码如果只有一句的话,是可以省略{ } 的
我们把Lam接口定义的调用方法参数修改一下,多出了一个String类型的形参s
//Lambda表达式调用的方法 void bda(String s);
这时候 我们如果使用Lambda表达式,则可以这样
//接口定义的方法有参数时候,并且你想要执行的操作也只有一句代码的时候 Lam lam1=e->System.out.println(e);//这一句还有简化版本 Lam lam1=System.out::println; lam1.bda("4556");
你又会发现,前面的()中括号也没了,这是因为当参数只有一个的时候,是可以省略()的。
当然也有你要执行很多代码的时候,那这时候可以这样
//接口定义的方法有参数时候,并且你想要执行的操作有很多句代码的时候 Lam lam1 = (String e) -> { String a = e + "add"; System.out.println(a); }; lam1.bda("test+");
输出结果如下
当然你还会问Lambda表达式能不能返回东西呢?这是肯定能的,首先我们再把上面的Lam接口方法修改一下
//Lambda表达式调用的方法 String bda(String s);
让bda方法返回一个String值,这次如果我们用Lambda的话
//接口定义的方法有返回值的时候 Lam lam1=s ->{System.out.println(s);return "我是返回的数据";}; lam1.bda("test1"); System.out.println(lam1.bda("test2"));
运行的结果:
总结 Lambda表达式 就是用来创建一个匿名的接口对象,即 它本身就是一个接口的匿名实例。只不过这个接口 有一些条件限制。
Lambda表达式的技巧
Lambda表达式只能用来简化仅包含一个public方法的接口的创建
规则
- 1.只能是接口
- 否则报:Target type of a lambda conversion must be an interface
- 2.只能有一个public方法
- 否则报:Multiple non-overriding abstract methods found AInterface
- 或AInterface is not a functional interface
括号形式
testA((int i, int j) -> {});参数要与接口一致
public class Go { public static void main(String a[]) { //正确示范 testA((int i, int j) -> {}); //错误示范:Multiple non-overriding abstract methods found xxx;只能有一个public方法 testB((int i, int j) -> {}); //错误示范:Target type of a lambda conversion must be an interface;只能是接口 testC((int i, int j) -> {}); } public static void testA(AInterface t) { } public static void testC(CInterface t) {} public static void testB(BInterface t) {} interface AInterface { void xxx(int i, int j); } interface BInterface { void xxx(int i, int j); void YYY(int i, int j); } abstract class CInterface { abstract void xxx(int i, int j); } }
双冒号表达形式
- 双冒号后面必须是静态方法
- 否则报错:Non-static method cannot be referenced from a static context
- 双冒号后面的方法与接口方法参数一样
- 方法与接口的权限可以不一样
- 返回类型:如果接口里面方法是void,双冒号后的方法可以任意返回类型,否则要一致
public class Go { public static void main(String a[]) { //之前的写法 testA(new AInterface() { @Override public void xxx(int i, int j) { } }); //正确,相对与接口里面xxx方这是改成静态和换了个名字 testA(Go::mydog); //正确,加了返回类型和public换成private,也是ok testA(Go::mydog2); //错误:Non-static method cannot be referenced from a static context testA(Go::mydog3); //这样写也是ok的。 AInterface aInterface = Go::mydog; testA(aInterface); } public static void testA(AInterface t) { t.xxx(1, 2); } interface AInterface { void xxx(int i, int j); } public static boolean mydog(int i, int j) { System.out.println("mydog" + i + " & " + j); return false; } private static void mydog2(int i, int j) { System.out.println("mydo2" + i + " & " + j); } public void mydog3(int i, int j) { System.out.println("mydog3" + i + " & " + j); } }
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。