一、Lambda 表达式的基础语法:Java8中引入了一个新的操作符 "->" 该操作符称为箭头操作符或 Lambda 操作符箭头操作符将 Lambda 表达式拆分成两部分:
左侧:Lambda 表达式的参数列表 右侧:Lambda 表达式中所需执行的功能, 即 Lambda 体 语法格式一:无参数,无返回值 () -> System.out.println("Hello Lambda!"); 语法格式二:有一个参数,并且无返回值 (x) -> System.out.println(x) 语法格式三:若只有一个参数,小括号可以省略不写 x -> System.out.println(x) 语法格式四:有两个以上的参数,有返回值,并且 Lambda 体中有多条语句 Comparatorcom = (x, y) -> { System.out.println("函数式接口"); return Integer.compare(x, y); }; 语法格式五:若 Lambda 体中只有一条语句, return 和 大括号都可以省略不写 Comparator com = (x, y) -> Integer.compare(x, y); 语法格式六:Lambda 表达式的参数列表的数据类型可以省略不写,因为JVM编译器通过上下文推断出,数据类型,即“类型推断” (Integer x, Integer y) -> Integer.compare(x, y);复制代码
二、Lambda 表达式需要“函数式接口”的支持 函数式接口:接口中只有一个抽象方法的接口,称为函数式接口。 可以使用注解 @FunctionalInterface 修饰 可以检查是否是函数式接口
三.Java8 内置的四大核心函数式接口
Consumer: 消费型接口 void accept(T t); Supplier : 供给型接口 T get(); Function : 函数型接口 R apply(T t); Predicate : 断言型接口 boolean test(T t);复制代码
其他内置函数式接口
函数式接口 | 参数类型 | 返回类型 | 用途 |
---|---|---|---|
BiFunction<T, U, R> | T, U | R | 对类型为 T, U 参数应用 操作,返回 R 类型的结 果。包含方法为 R apply(T t, U u); |
UnaryOperator (Function子接口) | T | T | 对类型为T的对象进行一 元运算,并返回T类型的 结果。包含方法为 T apply(T t); |
BinaryOperator (BiFunction 子接口) | T, T | T | 对类型为T的对象进行二 元运算,并返回T类型的 结果。包含方法为 T apply(T t1, T t2); |
BiConsumer<T, U> | T, U | void | 对类型为T, U 参数应用 操作。包含方法为 void accept(T t, U u) |
ToIntFunction ToLongFunction ToDoubleFunction | T | int long double | 分 别 计 算 int 、 long 、 double、值的函数 |
IntFunction LongFunction DoubleFunction | int long double | R | 参数分别为int、long、 double 类型的函数 |
package cn.lambda;import java.util.ArrayList;import java.util.Arrays;import java.util.List;import java.util.function.Consumer;import java.util.function.Function;import java.util.function.Predicate;import java.util.function.Supplier;import org.junit.Test;public class LambdaDemo { //Predicate断言型接口: @Test public void test4(){ List list = Arrays.asList("Hello", "Lambda", "www", "ok"); List strList = filterStr(list, (s) -> s.length() > 3); for (String str : strList) { System.out.println(str); } } //需求:将满足条件的字符串,放入集合中 public List filterStr(List list, Predicate pre){ List strList = new ArrayList<>(); for (String str : list) { if(pre.test(str)){ strList.add(str); } } return strList; } //Function 函数型接口: @Test public void test3(){ String newStr = strHandler("\t\t\t Demo ", (str) -> str.trim()); System.out.println(newStr); String subStr = strHandler("Demo1111", (str) -> str.substring(2, 5)); System.out.println(subStr); } //需求:用于处理字符串 public String strHandler(String str, Function fun){ return fun.apply(str); } //Supplier 供给型接口 : @Test public void test2(){ List numList = getNumList(10, () -> (int)(Math.random() * 100)); for (Integer num : numList) { System.out.println(num); } } //需求:产生指定个数的整数,并放入集合中 public List getNumList(int num, Supplier sup){ List list = new ArrayList<>(); for (int i = 0; i < num; i++) { Integer n = sup.get(); list.add(n); } return list; } //Consumer 消费型接口 : @Test public void test1(){ happy(10000, (m) -> System.out.println("demo:" + m + "元")); } public void happy(double money, Consumer con){ con.accept(money); }}可直接使用内置函数式接口,不满足时自定义接口即可复制代码
四、方法引用:若 Lambda 体中的功能,已经有方法提供了实现,可以使用方法引用 (可以将方法引用理解为 Lambda 表达式的另外一种表现形式)
1. 对象的引用 :: 实例方法名 2. 类名 :: 静态方法名 3. 类名 :: 实例方法名 注意: 1).方法引用所引用的方法的参数列表与返回值类型,需要与函数式接口中抽象方法的参数列表和返回值类型保持一致! 2).若Lambda 的参数列表的第一个参数,是实例方法的调用者,第二个参数(或无参)是实例方法的参数时,格式: ClassName::MethodName 二、构造器引用 :构造器的参数列表,需要与函数式接口中参数列表保持一致! 1. 类名 :: new 三、数组引用 类型[] :: new;复制代码
package cn.lambda;import java.io.PrintStream;import java.util.Comparator;import java.util.function.BiFunction;import java.util.function.BiPredicate;import java.util.function.Consumer;import java.util.function.Function;import java.util.function.Supplier;import org.junit.Test;public class ethodRef { //数组引用 @Test public void test8(){ Functionfun = (args) -> new String[args]; String[] strs = fun.apply(10); System.out.println(strs.length); Function fun2 = Employee[] :: new; Employee[] emps = fun2.apply(20); System.out.println(emps.length); } //构造器引用 @Test public void test7(){ Function fun = Employee::new; BiFunction fun2 = Employee::new; } @Test public void test6(){ Supplier sup = () -> new Employee(); System.out.println(sup.get()); Supplier sup2 = Employee::new; System.out.println(sup2.get()); } //类名 :: 实例方法名 @Test public void test5(){ BiPredicate bp = (x, y) -> x.equals(y); System.out.println(bp.test("abcde", "abcde")); BiPredicate bp2 = String::equals; System.out.println(bp2.test("abc", "abc")); Function fun = (e) -> e.show(); System.out.println(fun.apply(new Employee())); Function fun2 = Employee::show; System.out.println(fun2.apply(new Employee())); } //类名 :: 静态方法名 @Test public void test4(){ Comparator com = (x, y) -> Integer.compare(x, y); Comparator com2 = Integer::compare; } @Test public void test3(){ BiFunction fun = (x, y) -> Math.max(x, y); System.out.println(fun.apply(1.5, 22.2)); BiFunction fun2 = Math::max; System.out.println(fun2.apply(1.2, 1.5)); } //对象的引用 :: 实例方法名 @Test public void test2(){ Employee emp = new Employee(101, "张三", 18, 9999.99); Supplier sup = () -> emp.getName(); System.out.println(sup.get()); Supplier sup2 = emp::getName; System.out.println(sup2.get()); } @Test public void test1(){ PrintStream ps = System.out; Consumer con = (str) -> ps.println(str); con.accept("Hello World!"); Consumer con2 = ps::println; con2.accept("Hello Java8!"); Consumer con3 = System.out::println; } }复制代码