本节是lambda表达式的一个入门课,讲解的同时配有练习demo
前言什么是lambda表达式?基础语法函数式接口自己实现一个函数式接口jdk提供的函数式接口ConsumersupplierfunctionPredicate其他场景
lambda表达式是java8推出的一个新特性。标志着从jdk1.8开始,java正式支持函数式编程。使用lambda表达式可以用简洁流畅的代码完成一个功能,这也在一定程度上,可以减轻了java一直为人诟病的代码简洁性问题。
lambda表达式可以简单理解为一段可以传递的代码,核心思想是将面向对象中传递数据变为传递行为。下面举一个编写线程的例子。
1Runnable runnable = new Runnable() { @Override public void run() { System.out.println("hello lambda"); } 6}; 复制代码
这里使用的是匿名内部类的方式。匿名内部类实现的也是将代码作为数据传递,不过使用匿名内部类还是不够简洁,那么使用lambda表达式的代码是什么样的呢?
1Runnable runnable = () -> System.out.println("hello lambda"); 复制代码
我们可以看到使用lambda表达式实现起来非常的简单,只需要一行代码即可,接下来我们就试着分析一下这段代码。
我们可以看到上述的示例代码,lambda表达式的引用变量是一个runnable类型,事实上,java为我们提供了多种变量类型供我们选择,这些类型我们称之为函数式接口。
函数式接口是只有一个方法的接口,用作lambda表达式的类型。比如我们之前提到的Runnable接口
1@FunctionalInterface 2public interface Runnable { /** * When an object implementing interface <code>Runnable</code> is used * to create a thread, starting the thread causes the object's * <code>run</code> method to be called in that separately executing * thread. * <p> * The general contract of the method <code>run</code> is that it may * take any action whatsoever. * * @see java.lang.Thread#run() */ public abstract void run(); 15} 复制代码
我们可以看到interface上一个注解@FunctionInterface,它的作用是告诉编译器这个接口是一个函数式接口。
我们可以自己实现一个函数式接口,判断数据的数据是否合法
1/** * 自定义函式数接口 * * @author jialin.li * @date 2020-03-18 14:22 */ 7@FunctionalInterface 8public interface CustomFunctionInterface<T> { boolean legal(T t); 10} 复制代码
测试方法:
1/** * 测试. * * @author jialin.li * @date 2020-03-18 14:23 */ 7public class Test { public boolean legal(int value, CustomFunctionInterface<Integer> functionInterface){ return functionInterface.legal(value); } public static void main(String[] args) { System.out.println(new Test().legal(-8, (x) -> x >= 0)); } 16} 复制代码
测试结果:
1false 复制代码
jdk在java.util.function中为我们提供了非常丰富的函数式接口,大致可以分为4类:
将输入的10进制数,转换为2进制数输出
1public void toBinaryString(int value, Consumer<Integer> consumer){ consumer.accept(value); 3} 5public static void main(String[] args) { new Test().toBinaryString(15, (x) -> System.out.println(Integer.toBinaryString(x))); 7} 复制代码
生成n个随机数100以内的随机数,以数组的形式返回
1public int[] getRandomArr(int n, Supplier<Integer> supplier) { int[] arr = new int[n]; for (int i = 0; i < n; i++) { arr[i] = supplier.get(); } return arr; 7} 9public static void main(String[] args) { int[] arr = new Test().getRandomArr(10, () -> new Random().nextInt(100)); System.out.println(Arrays.toString(arr)); 12} 复制代码
生成一个[1,n]组成的阶乘数组,例如输入3,返回[1,4,9]
1public int[] getFactorialArr(int n, Function<Integer,Integer> function) { int[] arr = new int[n]; for (int i = 1; i <= n; i++) { arr[i - 1] = function.apply(i); } return arr; 7} 9public static void main(String[] args) { int[] arr = new Test().getFactorialArr(10, (x) -> x*x); System.out.println(Arrays.toString(arr)); 12} 复制代码
判断一个数是否是偶数
1public boolean isEven(int n, Predicate<Integer> predicate) { return predicate.test(n); 3} 5public static void main(String[] args) { boolean isEven = new Test().isEven(1102, (x) -> (x % 2 == 0)); System.out.println(isEven); 8} 复制代码
除了上述的4个接口,java.util.function中还有许多其他的接口,可以实现各种功能。比如实现2个数的加法运算:
1public static void main(String[] args) { BinaryOperator<Integer> binaryOperator = Integer::sum; System.out.println(binaryOperator.apply(1, 3)); 4} 复制代码
实现List遍历:
1public static void main(String[] args) { List<String> list = new ArrayList<>(Arrays.asList("java", "python", "go", "sql")); list.forEach(System.out::println); 4} 复制代码
更多场景还请您去探索,感受函数式编程的魅力吧
最后,期待您的订阅和点赞,专栏每周都会更新,希望可以和您一起进步,同时也期待您的批评与指正!