Java1.8之Lambda表达式

1、Java8的lambda表达式,通过lambda表达式可以替代我们之前写的匿名内部类来实现接口。lambda表达式本质是一个匿名函数。

 1 package com.demo.main;
 2 
 3 public class LambdaMain {
 4 
 5     public static void main(String[] args) {
 6 
 7         // 1、创建一个匿名内部类
 8         Addition addition = new Addition() {
 9 
10             @Override
11             public int add(int a, int b) {
12 
13                 return a + b;
14             }
15 
16         };
17         // 传统的匿名内部类来实现接口。
18         System.out.println("调用匿名内部类来实现接口:" + addition.add(2, 3));
19 
20         // 2、Java8的lambda表达式,通过lambda表达式可以替代我们之前写的匿名内部类来实现接口。
21         // lambda表达式本质是一个匿名函数。
22         // 2.1、lambda由三部分组成,()是参数列表,->剪头符号,{}代表方法体。
23         Addition a2 = (int a, int b) -> {
24             return a + b;
25         };
26         System.out.println("lambda表达式a2:" + a2.add(3, 3));
27 
28         // 2.2、可以省略参数列表里面的参数类型
29         Addition a3 = (a, b) -> {
30             return a + b;
31         };
32         System.out.println("lambda表达式a3:" + a3.add(3, 3));
33 
34     }
35 
36     /**
37      * 
38      * @author 创建一个接口,定义一个方法
39      *
40      */
41     interface Addition {
42 
43         /**
44          * 加法的方法
45          * 
46          * @param a
47          * @param b
48          * @return
49          */
50         int add(int a, int b);
51     }
52 
53 }

 

2、Lambda表达式语法,形如(int a, int b) -> {return a + b;},lambda本质就是一个(匿名)函数,匿名函数,就是没有方法名称的函数。

1 一般函数结构类似,如下所示:
2 int add(int a, int b){
3     return  a + b;
4 }
5 
6 一般函数的,结构如:返回值 方法名称(参数列表) 方法体。

 

3、而Lamdba表达式函数,只有参数列表和方法体。结构如:(参数列表) -> {方法体};

详细说明,如下所示:

  1)、()括号用来描述参数列表。
  2)、{}大括号用来描述方法体。
  3)、->尖括号,Lambda运算符,可以叫做箭头符号,或者goes to。

 

4、Lambda表达式语法,关于接口方法参数、无参、单个参数、两个参数、有返回值、没有返回值的情况。如何来根据lambda来返回接口函数。

  1 package com.demo.main;
  2 
  3 public class LambdaMain {
  4 
  5     public static void main(String[] args) {
  6         // 1、无参数无返回值
  7         MethodNoReturnNoParam methodNoReturnNoParam = () -> {
  8             System.out.println("无参数无返回值");
  9         };
 10         methodNoReturnNoParam.lambda1();
 11 
 12         // 2、单个参数无返回值
 13         MethodNoReturnOneParam methodNoReturnOneParam = (int a) -> {
 14             System.out.println("单个参数无返回值,a = " + a);
 15         };
 16         methodNoReturnOneParam.lambda2(3);
 17 
 18         // 3、两个参数无返回值
 19         MethodNoReturnTwoParam methodNoReturnTwoParam = (int a, int b) -> {
 20             System.out.println("两个参数无返回值,a + b = " + (a + b));
 21         };
 22         methodNoReturnTwoParam.lambda3(3, 4);
 23 
 24         // 4、无参数有返回值
 25         MethodReturnNoParam methodReturnNoParam = () -> {
 26             return 8;
 27         };
 28         System.out.println("无参数有返回值:" + methodReturnNoParam.lambda4());
 29 
 30         // 5、一个参数有返回值
 31         MethodReturnOneParam methodReturnOneParam = (int a) -> {
 32             return a;
 33         };
 34         System.out.println("一个参数有返回值,a = " + methodReturnOneParam.lambda5(9));
 35 
 36         // 6、一个参数有返回值
 37         MethodReturnTwoParam methodReturnTwoParam = (int a, int b) -> {
 38             return a + b;
 39         };
 40         System.out.println("一个参数有返回值,a + b = " + methodReturnTwoParam.lambda6(9, 9));
 41 
 42     }
 43 
 44     /**
 45      * 1、无参数无返回值
 46      * 
 47      * @author
 48      *
 49      */
 50     interface MethodNoReturnNoParam {
 51 
 52         void lambda1();
 53     }
 54 
 55     /**
 56      * 2、单个参数无返回值
 57      * 
 58      * @author
 59      *
 60      */
 61     interface MethodNoReturnOneParam {
 62 
 63         void lambda2(int a);
 64     }
 65 
 66     /**
 67      * 3、两个参数无返回值
 68      * 
 69      * @author
 70      *
 71      */
 72     interface MethodNoReturnTwoParam {
 73 
 74         void lambda3(int a, int b);
 75     }
 76 
 77     /**
 78      * 4、无参数有返回值
 79      * 
 80      * @author
 81      *
 82      */
 83     interface MethodReturnNoParam {
 84 
 85         int lambda4();
 86     }
 87 
 88     /**
 89      * 5、一个参数有返回值
 90      * 
 91      * @author
 92      *
 93      */
 94     interface MethodReturnOneParam {
 95 
 96         int lambda5(int a);
 97     }
 98 
 99     /**
100      * 6、两个参数有返回值
101      * 
102      * @author
103      *
104      */
105     interface MethodReturnTwoParam {
106 
107         int lambda6(int a, int b);
108     }
109 
110 }

 

5、Lambda表达式语法,精简写法,如下所示:

  1)、参数类型可以省略。
  2)、假如只有一个参数,()括号可以省略。
  3)、如果方法体只有一条语句,{}大括号可以省略。
  4)、如果方法体中唯一的语句是return返回语句,那省略大括号的同时return关键词也要省略掉。

 1 package com.demo.main;
 2 
 3 public class LambdaMain {
 4 
 5     public static void main(String[] args) {
 6         // 1、无参数无返回值
 7         MethodNoReturnNoParam methodNoReturnNoParam = () -> System.out.println("无参数无返回值");
 8         methodNoReturnNoParam.lambda1();
 9 
10         // 2、单个参数无返回值
11         MethodNoReturnOneParam methodNoReturnOneParam = a -> System.out.println("单个参数无返回值,a = " + a);
12         methodNoReturnOneParam.lambda2(3);
13 
14         // 3、两个参数无返回值
15         MethodNoReturnTwoParam methodNoReturnTwoParam = (a, b) -> System.out.println("两个参数无返回值,a + b = " + (a + b));
16         methodNoReturnTwoParam.lambda3(3, 4);
17 
18         // 4、无参数有返回值
19         MethodReturnNoParam methodReturnNoParam = () -> 8;
20         System.out.println("无参数有返回值:" + methodReturnNoParam.lambda4());
21 
22         // 5、一个参数有返回值
23         MethodReturnOneParam methodReturnOneParam = a -> a + 10;
24         System.out.println("一个参数有返回值,a = " + methodReturnOneParam.lambda5(9));
25 
26         // 6、一个参数有返回值
27         MethodReturnTwoParam methodReturnTwoParam = (a, b) -> a + b;
28         System.out.println("一个参数有返回值,a + b = " + methodReturnTwoParam.lambda6(9, 9));
29 
30     }
31 
32     /**
33      * 1、无参数无返回值
34      * 
35      * @author
36      *
37      */
38     interface MethodNoReturnNoParam {
39 
40         void lambda1();
41     }
42 
43     /**
44      * 2、单个参数无返回值
45      * 
46      * @author
47      *
48      */
49     interface MethodNoReturnOneParam {
50 
51         void lambda2(int a);
52     }
53 
54     /**
55      * 3、两个参数无返回值
56      * 
57      * @author
58      *
59      */
60     interface MethodNoReturnTwoParam {
61 
62         void lambda3(int a, int b);
63     }
64 
65     /**
66      * 4、无参数有返回值
67      * 
68      * @author
69      *
70      */
71     interface MethodReturnNoParam {
72 
73         int lambda4();
74     }
75 
76     /**
77      * 5、一个参数有返回值
78      * 
79      * @author
80      *
81      */
82     interface MethodReturnOneParam {
83 
84         int lambda5(int a);
85     }
86 
87     /**
88      * 6、两个参数有返回值
89      * 
90      * @author
91      *
92      */
93     interface MethodReturnTwoParam {
94 
95         int lambda6(int a, int b);
96     }
97 
98 }

 

6、方法引用,如果多个lamdba表达式实现函数都是一样的话,可以封装成通用方法,以方便维护,此时可以使用方法引用实现。

  语法规则:对象::方法。假如是static静态方法,可以直接类名::方法。

 1 package com.demo.main;
 2 
 3 public class LambdaMain {
 4 
 5     public static void main(String[] args) {
 6 //        // 创建对象
 7 //        LambdaMain lambdaMain = new LambdaMain();
 8 //        // 采用对象引用的方式进行实现
 9 //        MethodReturnOneParam methodReturnOneParam_1 = lambdaMain::add;
10 //        System.out.println(methodReturnOneParam_1.lambda5(10));
11 //
12 //        // 采用对象引用的方式进行实现
13 //        MethodReturnOneParam methodReturnOneParam_2 = lambdaMain::add;
14 //        System.out.println(methodReturnOneParam_2.lambda5(40));
15         
16         
17         // 创建对象
18         // 采用静态对象引用的方式进行实现
19         MethodReturnOneParam methodReturnOneParam_1 = LambdaMain::addStatic;
20         System.out.println(methodReturnOneParam_1.lambda5(10));
21 
22         // 采用静态对象引用的方式进行实现
23         MethodReturnOneParam methodReturnOneParam_2 = LambdaMain::addStatic;
24         System.out.println(methodReturnOneParam_2.lambda5(40));
25     }
26 
27     /**
28      * 方法引用,如果多个lamdba表达式实现函数都是一样的话,可以封装成通用方法,以方便维护,此时可以使用方法引用实现。
29      * 
30      * @param a
31      * @return
32      */
33     public int add(int a) {
34         return a + 10;
35     }
36 
37     /**
38      * 静态方法引用
39      * 
40      * @param a
41      * @return
42      */
43     public static int addStatic(int a) {
44         return a + 10;
45     }
46 
47     /**
48      * 5、一个参数有返回值
49      *
50      * @author
51      *
52      */
53     interface MethodReturnOneParam {
54 
55         int lambda5(int a);
56     }
57 
58 }

 

7、构造方法引用,如果函数式接口的实现切好可以通过调用一个类的构造方法来实现,那么就可以使用构造方法引用。

语法规则:类名::new。

 1 package com.demo.po;
 2 
 3 public class Dog {
 4 
 5     private String name;
 6     private int age;
 7 
 8     public String getName() {
 9         return name;
10     }
11 
12     public void setName(String name) {
13         this.name = name;
14     }
15 
16     public int getAge() {
17         return age;
18     }
19 
20     public void setAge(int age) {
21         this.age = age;
22     }
23 
24     @Override
25     public String toString() {
26         return "Dog [name=" + name + ", age=" + age + "]";
27     }
28 
29     public Dog(String name, int age) {
30         System.out.println("含参构造函数!");
31         this.name = name;
32         this.age = age;
33     }
34 
35     public Dog() {
36         System.out.println("无参构造函数!");
37     }
38 
39 }

 

 1 package com.demo.main;
 2 
 3 import com.demo.po.Dog;
 4 
 5 public class LambdaMain {
 6 
 7     public static void main(String[] args) {
 8         // 1、使用lambda表达式的方式使用
 9         DogService dogService1_0 = () -> {
10             return new Dog();
11         };
12         dogService1_0.getDog();
13 
14         // 2、简化方式
15         DogService dogService1_1 = () -> new Dog();
16         dogService1_1.getDog();
17 
18         // 3、构造方法引用,如果函数式接口的实现切好可以通过调用一个类的构造方法来实现,那么就可以使用构造方法引用。
19         DogService dogService1_2 = Dog::new;
20         dogService1_2.getDog();
21 
22         // 4、构造方法引用,有参构造函数调用
23         DogService2 dogService2_1 = Dog::new;
24         dogService2_1.getDog("小黄", 2);
25 
26     }
27 
28     interface DogService {
29 
30         /**
31          * 获取到一只无名狗
32          * 
33          * @return
34          */
35         Dog getDog();
36     }
37 
38     interface DogService2 {
39 
40         /**
41          * 获取到一个有名称的狗
42          * 
43          * @param name
44          * @param age
45          * @return
46          */
47         Dog getDog(String name, int age);
48     }
49 
50 }

 

8、集合使用lambda表达式对数据进行排序,遍历等操作。对于参数是接口函数的方法,需要传递lamdba表达式作为参数进行调用。

 1 package com.demo.main;
 2 
 3 import java.util.ArrayList;
 4 import java.util.List;
 5 
 6 import com.demo.po.Dog;
 7 
 8 public class LambdaMain {
 9 
10     public static void main(String[] args) {
11         List<Dog> list = new ArrayList<Dog>();
12         list.add(new Dog("小黄", 5));
13         list.add(new Dog("小花", 4));
14         list.add(new Dog("旺财", 3));
15         list.add(new Dog("团团", 2));
16         list.add(new Dog("圆圆", 1));
17 
18         // 排序
19         System.out.println("lambda集合排序!!!");
20         // sort的参数是接口函数,需要使用lambda表达式匿名函数形式
21         list.sort((dog1, dog2) -> dog1.getAge() - dog2.getAge());
22         System.out.println(list);
23         System.out.println();
24 
25         // 遍历集合
26         System.out.println("lamdba集合遍历:");
27         list.forEach(System.out::println);
28     }
29 
30 }

 

9、@FunctionalInterface注解,此注解是函数式接口注解,所谓的函数式接口,当然首先是一个接口,然后就是在这个接口里面只能有一个抽象方法。这种类型的接口也称为SAM接口,即Single Abstract Method interfaces。

特点:

  1)、接口有且仅有一个抽象方法。
  2)、运行定义静态方法。
  3)、允许定义默认方法。
  4)、允许java.lang.Object中的public方法。
  5)、该注解不是必须的,如果一个接口符合"函数式接口"定义,那么加不加该注解都没有影响。加上该注解能够更好的让编译器进行检查。如果编写的不是函数时接口,但是加上该注解,那么编译器会报错。

 1 package com.demo.main;
 2 
 3 public class LambdaMain {
 4 
 5     public static void main(String[] args) {
 6         FunctionInterface functionInterface = (int a) -> {
 7             System.out.println("a = " + a);
 8         };
 9         functionInterface.add(10);
10     }
11 
12     /**
13      * 正确的函数式接口
14      * 
15      * @author biexiansheng
16      *
17      */
18     @FunctionalInterface
19     interface FunctionInterface {
20 
21         /**
22          * 抽象方法
23          */
24         public void add(int a);
25 
26         /**
27          * java.lang.Object中的public方法
28          * 
29          * @param var
30          * @return
31          */
32         public boolean equals(Object var);
33 
34         // 默认的方法
35         public default void defaultMethod() {
36             System.out.println("默认的方法!!!");
37         }
38 
39         // 静态方法
40         public static void staticMethod() {
41             System.out.println("静态的方法!!!");
42         }
43 
44     }
45 
46 }

 

10、系统内置函数式接口,Java8的推出,是以lamdba重要特性,一起推出的,其中系统内置了一系列函数式接口。在jdk的java.util.function包下,有一系列的内置函数式接口:

 1 package com.demo.main;
 2 
 3 import java.util.function.IntConsumer;
 4 import java.util.function.IntFunction;
 5 import java.util.function.IntPredicate;
 6 
 7 public class LambdaMain {
 8 
 9     public static void main(String[] args) {
10         // 在jdk的java.util.function包下,有一系列的内置函数式接口
11         // 1、使用int函数接口
12         IntFunction<Integer> intFunction = (a) -> {
13             return a + 10;
14         };
15         System.out.println(intFunction.apply(20));
16         
17         // 2、使用int判断函数接口
18         final int aa = 20;
19         IntPredicate intPredicate = (a) -> {
20             return a == aa;
21         };
22         System.out.println(intPredicate.test(aa));
23         
24         // 3、使用int传递参数的形式
25         IntConsumer intConsumer = (a) -> {
26             System.out.println("a = " + a);
27         };
28         intConsumer.accept(20);
29         
30         // 书写技巧:首先定义一个接口函数对象,然后后面跟lambda表达式。
31         // lambda参数列表根据接口函数的方法参数类型和参数个数。
32         // lambda的方法体,是否有返回值,根据接口函数的方法是否有返回值。
33     }
34 
35 }

 

posted on 2021-01-09 16:54  别先生  阅读(833)  评论(0编辑  收藏  举报