java表达式由什么组成_必知必会之Lambda表达式

Java是一门强大的面向对象的语言,除了8种基本的数据类型,其他一切皆为对象。因此,在Java中定义函数或方法都离不开对象,也就意味着很难直接将方法或函数像参数一样传递,而Java8中的Lambda表达式解决了这个问题。

一、为什么需要Lambda?

简单的来说,引入Lambda就是为了简化代码,允许把函数作为一个方法的参数传递进方法中。

1.1 真的简化了?

示例:如果想把某个接口的实现类作为参数传递给一个方法会怎么做?

Java8以前

public static void general() {

// 用匿名内部类的方式来创建线程

new Thread(new Runnable() {

@Override

public void run() {

System.out.println("公众号:风尘博客!");

}

}).run();

}

Lambda 写法

public static void lambda() {

// 使用Lambda来创建线程

new Thread(() -> System.out.println("公众号:风尘博客!")).run();

}

1.2 Lambda表达式是什么?

Java中,将方法作为参数进行传递的方式被称为Lambda表达式。

1.3 Lambda 表达式语法结构

Lambda其实是一个箭头函数,也可称为匿名函数:->

箭头操作符将Lambda表达式分成了两部分:

左侧:Lambda表达式的参数列表(接口中抽象方法的参数列表)

右侧:Lambda表达式中所需执行的功能(Lambda体,对抽象方法的实现)

1.4 语法格式

无参,无返回值,Lambda 体只需一条语句。

public static void noParam() {

Runnable r1 = () -> System.out.println("noParam Test!");

r1.run();

}

Lambda 需要一个参数,参数的小括号可以省略。

public static void oneParam() {

// Consumer con = (s) -> System.out.println(s);

// 参数的小括号可以省略。

Consumer con = s -> System.out.println(s);

con.accept("oneParam Test!");

}

Lambda 需要多个参数,并且有返回值。

public static void params() {

Comparator com = (x, y) -> {

System.out.println("函数式接口");

// 比较x/y的大小

return Integer.compare(x, y);

};

System.out.println(com.compare(1, 2));

}

当 Lambda 体只有一条语句时,return 与大括号可以省略。

public static void one() {

Comparator com = (x, y) -> Integer.compare(x, y);

System.out.println(com.compare(1, 2));

}

上面几条示例好像有一个共性:参数列表的数据类型都没写,这是为什么呢?

1.5 类型推断

Lambda 表达式中的参数类型都是由编译器推断得出的。

public static void typeInference() {

//Integer 类型可以省略

Comparator com = (Integer x,Integer y) -> {

System.out.println("函数式接口");

return Integer.compare(x, y);

};

// 类型推断

BinaryOperator addImplicit = (x, y) -> x + y;

}

Lambda 表达式中无需指定类型,程序依然可 以编译,这是因为 javac根据程序的上下文,在后台 推断出了参数的类型。Lambda 表达式的类型依赖于上下文环境,是由编译器推断出来的。

1.6 小节

Lambda表达式使得Java拥有了函数式编程的能力,但在Java中Lambda表达式是对象,它必须依附于一类特别的对象类型——函数式接口(functional interface)。

二、函数式接口

函数接口是只有一个抽象方法的接口,用作 Lambda 表达式的类型。使用@FunctionalInterface注解修饰的类,编译器会检测该类是否只有一个抽象方法或接口,否则,会报错。可以有多个默认方法,静态方法。

JDK8在 java.util.function 中定义了几个标准的函数式接口,供我们使用。

2.1 Java 内置四大核心函数式接口

函数式接口

参数类型

返回类型

用途

Consumer

T

void

对类型为T的对象应用操作,包含方法:void accept(T t)

Supplier

T

返回类型为T的对象,包 含方法:T get();

Function

T

R

对类型为T的对象应用操作,并返回结果。结果是R类型的对象。包含方法:R apply(T t);

Predicate

T

boolean

确定类型为T的对象是否满足某约束,并返回 boolean 值。包含方法 boolean test(T t);

消费型接口

void accept(T t);

consumerDemo(3, s -> System.out.println(s * 3));

public static void consumerDemo(Integer value, Consumer consumer) {

consumer.accept(value);

}

供给型接口

T get();

// 生成10个以内的随机书

List numList = supplierDemo(10, () -> (int)(100 * Math.random()));

System.out.println(numList);

public static List supplierDemo(int num, Supplier supplier) {

List list = new ArrayList<>();

for (int i = 0; i < num; i++) {

Integer n = supplier.get();

list.add(n);

}

return list;

}

函数型接口

R apply(T t);

// 处理字符串

String str1 = functionDemo("Hello!风尘博客", s -> s.substring(6));

System.out.println(str1);

String str2 = functionDemo("vanDusty", s -> s.toUpperCase());

System.out.println(str2);

public static String functionDemo(String str, Function function) {

return function.apply(str);

}

断言型接口

boolean test(T t);

// 将满足条件的字符串放入集合

List list = Arrays.asList("hello", "van", "function", "predicate");

List newList = predicateDemo(list, s -> s.length() > 5);

System.out.println(newList);

public static List predicateDemo(List list, Predicate predicate) {

List newList = new ArrayList<>();

for (String s : list) {

if (predicate.test(s)) {

newList.add(s);

}

}

return newList;

}

2.2 自定义函数式接口

我们可以在任意函数式接口上使用 @FunctionalInterface 注解, 这样做可以检查它是否是一个函数式接口,同时 javadoc 也会包含一条声明,说明这个接口是一个函数式接口。

// 字符串转大写

String newStr = selfFunctionalInterface((str) -> str.toUpperCase(), "abc");

System.out.println(newStr);

public static String selfFunctionalInterface(SelfFunctionalInterface selfFunctionalInterface, String str) {

return selfFunctionalInterface.getValue(str);

}

三、方法引用和构造器引用

3.1 方法引用

方法引用是指通过方法的名字来指向一个方法。

3.1.1 方法引用使用的前提条件是什么呢?

方法引用所引用的方法的参数列表必须要和函数式接口中抽象方法的参数列表相同(完全一致);

方法引用所引用的方法的的返回值必须要和函数式接口中抽象方法的返回值相同(完全一致)。

3.1.2 方法引用三种格式

实例对象名::实例方法名

private static void instanceMethod() {

UserDomain user = new UserDomain(1L, "Van");

Supplier sup = () -> user.getUserName();

System.out.println(sup.get());

// 等同于

Supplier supplier = user::getUserName;

System.out.println(supplier.get());

}

类名::静态方法名

private static void staticMethod() {

Comparator com = (x, y) -> Integer.compare(x, y);

System.out.println(com.compare(3,9));

// 等同于

Comparator com2 = Integer::compare;

System.out.println(com2.compare(3,9));

}

类名::实例方法名

private static void instanceMethodObject() {

UserDomain user = new UserDomain(1L, "Van");

Function fun = (e) -> e.getUserName();

System.out.println(fun.apply(user));

// 等同于

Function fun2 = UserDomain::getUserName;

System.out.println(fun2.apply(user));

}

3.2 构造器引用

前提:构造器参数列表要与接口中抽象方法的参数列表一致!

语法格式:类名 :: new

构造器引用

private static void object() {

// UserDomain 中必须有一个 UserDomain(String userName) 的构造器,下同

Function fun = (n) -> new UserDomain(n);

fun.apply("Van");

System.out.println("===等价于===");

Function function = UserDomain::new;

function.apply("Van");

// 带两个参数的构造器引用就要用BiFunction,多个参数的话,还可以自定义一个这样的函数式接口

BiConsumer biConsumer = UserDomain :: new;

biConsumer.accept(1L,"Van");

}

数组引用

private static void array() {

//传统Lambda实现

Function function = (i) -> new int[i];

int[] apply = function.apply(10);

System.out.println(apply.length);

//数组类型引用实现

function = int[] ::new;

apply = function.apply(100);

System.out.println(apply.length);

}

四、 总结

Lambda表达式是Java对于函数式编程的温和转变,面向对象编程和函数式编程不是互相对立的,结合使用能够更加有效地帮助我们管理程序的复杂性。

技术交流

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值