lambda表达式
lambda表达式是一个匿名函数,我们可以把 Lambda 表达式理解为是 一段可以传递的代码(将代码像数据一样进行传递)。使用它可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升。
但是在java中,是没有函数这个概念的。另外,依旧Java“万事万物皆对象”思想,java将lambda表达式做为一个接口(函数式接口,下面会说明)的实例。
首先,我们来感受下lambda表达式的简洁。
案例:匿名实现类和lambda表达式的对比
Runnable run1 = new Runnable(){
@Override
public void run() {
int sum = 0;
for (int i = 1; i<=100; i++) {
sum += i;
}
System.out.println("100以内所有自然是的和为:" + sum);
}
};
run1.run();
System.out.println("**********************************************************");
Runnable run2 = () -> {
int sum = 0;
for (int i = 1; i<=100; i++) {
sum += i;
}
System.out.println("100以内所有自然是的和为:" + sum);
};
run2.run();
用匿名实现类表示的都可以用Lambda表达式来写,所以对于使用了匿名实现类的地方,最好都用lambda表达式重构下,更加简洁,另外逼格更高哦!
lambda表达式语法
基本语法格式如下:
(参数列表)->{方法体};
一般格式如下:
(String s1, String s2) -> {
if (s1.length() > s2.length()) {
return 1;
} else if (s1.length() < s2.length()) {
return -1;
} else {
return 0;
}
};
如果没有参数,左边的()也不可以省略。
() => {System.out.println("hello wrold!");};
如果可以推到出一个lambda表达式的参数类型,则可以忽略其类型。
(str)->{System.out.printnln(str);};
格式四,如果只有一个参数且能推导出其类型,可省略括号。
str->{System.out.println(str);};
格式五,如果右边只有一条语句时,可以省略大括号以及return
(str1,str2) -> str1.length() - str2.length();
函数式接口
只包含一个抽象方法的接口,称为函数式接口。
在函数式编程语言当中,函数被当做一等公民对待。在将函数作为一等公民的编程语言中,Lambda表达式的类型是函数。但是在Java8中,有所不同。在Java8中,Lambda表达式是对象,而不是函数,它们必须依附于一类特别的对象类型——函数式接口。
简单的说,在Java8中,Lambda表达式就是一个函数式接口的实例。这就是Lambda表达式和函数式接口的关系。也就是说,只要一个对象是函数式接口的实例,那么该对象就可以用Lambda表达式来表示。
Comparator<String> com = (str1,str2) -> str1.length() - str2.length();
我们可以在一个接口上使用 @FunctionalInterface 注解,这样做可以检查它是否是一个函数式接口。同时 javadoc 也会包含一条声明,说明这个接口是一个函数式接口。
@FunctionalInterface
interface MyInterface
{
int method1 (int n1);
}
public class T2 {
public static void main(String[] args) {
MyInterface i = n1->n1 * 3;
int n = i.method1(100);
System.out.println(n);
}
}
常用函数式接口
使用基本类型的函数式接口,通常效率更高,推荐使用。下面是常见基本类型的函数式接口:
package com.studyjava.demo;
import java.util.function.*;
import java.util.*;
public class Demo14
{
public static void main (String[] args)
{
testConsumer("this is a test code about testConsumer!", str->System.out.println(str));
testSupplier(()
->(int) (Math.random() * 10 + 1)
);
testFunction("java is a good language", str->str.length());
testPredicate(new int[]{3,-2,43,-1}, n->n>0);
}
public static void testConsumer (String str, Consumer<String> con)
{
con.accept(str);
}
public static void testSupplier (Supplier<Integer> sup)
{
System.out.println("(1~10)random num is " + sup.get());
}
public static void testFunction (String str, Function<String, Integer> fun)
{
int n = fun.apply(str);
System.out.println(str + " length is " + n);
}
public static void testPredicate (int[] arrs, Predicate<Integer> pre)
{
var arrlist = new ArrayList<Integer>();
for (int n : arrs) {
if (pre.test(n)) {
arrlist.add(n);
}
}
for (int i:arrlist) {
System.out.println(i);
}
}
}