函数式编程/lambda表达式入门
函数式编程/lambda表达式入门
本篇主要讲解 lambda表达式的入门,涉及为什么使用函数式编程,以及jdk8提供的函数式接口 和 接口的默认方法 等等
1.什么是命令式编程
命令式编程就是我们去告诉程序如何实现
比如在一堆数字中寻找最小值
int[] nums = {33, 44, 55, -111, -1};
int minNum = Integer.MAX_VALUE;
for (int num : nums) {
if (num < minNum) {
minNum = num;
}
}
System.out.println(minNum);
这段代码就是命令式编程,我们去告诉程序如何从一堆数字中找到最小值
1.什么是函数式编程
它不是一种具体的技术,是一种方法论,使用对应提供的函数进行编程,而不是去关心内部的实现
比如在一堆数字中寻找最小值
int[] nums = {33, 44, 55, -111, -1};
int min = IntStream.of(nums).min().getAsInt();
System.out.println(min);
IntStream.of(nums).min() 就是面向函数式编程,你不需要去实现具体的逻辑,只需要简单调用
2.函数式编程的好处
2.1 易于并发编程 parallel()
比如在一堆数字中寻找最小值
当这堆数字很大的时候,我们需要去实现多线程去比较的话,需要考虑多线程,线程池以及拆分数据,实现快排等。。但是当我们使用JDK8函数式编程的时候可以直接使用一个方法开启 并行 .parallel()
int min = IntStream.of(nums).parallel().min().getAsInt();
2.2 代码简洁
可以明显看出来函数式编程代码简洁了很多
int[] nums = {33, 44, 55, -111, -1};
函数式编程:int min = IntStream.of(nums).parallel().min().getAsInt();
命令式编程:
int minNum = Integer.MAX_VALUE;
for (int num : nums) {
if (num < minNum) {
minNum = num;
}
}
System.out.println(minNum);
2.3 等等。。。
3.lambda 表达式初接触
在Java中我们创建线程会使用这种方式,弄一个匿名内部类去实现Runnable接口,但是线程执行的核心代码 只是run方法里面的 但是我们需要用这么多行代码去实现,实在不是很友好
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("lambda 初识");
}
}).start();
lambda表达式的方式 实现创建一个线程
new Thread(() -> System.out.println("lambda 初识")).start();
可以看出来比上面匿名内部类要简单很多
在Java8中用使用 -> 函数来编写lambda表达式, lambda表达式会返回一个函数
Runnable runnable = () -> System.out.println("lambda 初识");
lambda表达式不关心你要实现的是什么接口,只是关心你的 参数 和返回值 ,如下
interface MyFunction{
void hello();
}
MyFunction myFunction = () -> System.out.println("lambda 初识");
4.JDK8中的函数式接口
4.1 概念:
函数式接口是JDK8中引入的概念,对应的注解是 @FunctionalInterface 标注在接口上表示这个接口只有一个方法(不算默认方法) 如下
@FunctionalInterface
interface Interface1 {
int doubleNumber(int i);
}
4.2 @FunctionalInterface
@FunctionalInterface 这个注解只是起到编译时的检查,不加也行 但是推荐要加上
Interface1 interface1 = (i) -> i * 2;
当我们在标注了@FunctionalInterface这个注解的接口里添加了 2个或者多个方法的时候 会产生编译错误
@FunctionalInterface
interface Interface1 {
int doubleNumber(int i);
void someMethod();
}
当我们仔细观察 会发现lambda 表达式只是关注的入参和返回值,也就是说可以把这个lambda表达式赋给任何的有一个参数有一个返回值的函数式接口上
Interface1 interface1 = (i) -> i * 2;
这时候就引入了JDK8提供的默认的函数式接口,大多数情况下就不用我们自己去定义接口了
4.3 默认函数式接口
使用默认的Function函数式接口
Function<Integer , Integer> function = (i) -> i * 2;
function.apply(10);
使用默认的Predicate函数式接口
Predicate<Integer> predicate = (i) -> i % 2 == 0;
predicate.test(2);
5.JDK8接口新特性 默认方法
JKD8接口新特性 默认方法,这个新特性其实是很重要的,它的作用就是在接口上可以定义方法的实现 作为默认方法,这样 Java就可以扩展很多底层接口 比如List接口 ,对其添加了很多default默认方法,这时候List的接口的实现类也不用去修改了
前面有介绍关于@FunctionalInterface 会检测该接口有几个待实现的方法,如果有多个则编译不通过,但是 当我们在另一个方法上面添加 default
@FunctionalInterface
interface Interface1 {
int doubleNumber(int i);
void someMethod();
}
-》 如下添加default ,此时接口Interface1 还是保持一个待实现的接口,此时@FunctionalInterface编译就能通过
@FunctionalInterface
interface Interface1 {
int doubleNumber(int i);
default void someMethod(){
System.out.println("Interface1 default Method ");
}
}
如List提供了 一个sort的default方法,这是以前List接口没有的
6.总结
本篇主要讲解 lambda表达式的入门,涉及函数式编程和命令式编程的区别,以及jdk8提供的函数式接口 和 接口的默认方法 等等该篇只涉及入门概念,具体深入理解还是需要自己去实践,最近准备学习Webflux 所以需要复习一下Java8 lambda 以及 Stream流