Java8新特性之lambda表达式,函数式接口,方法、构造器、数组引用(附代码案例)

1. lambda表达式

1. 为什么使用Lambda表达式?

Lambda表达式是一个匿名函数,我们可以把Lambda表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。使用它可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使Java的语言表达能力得到了提升

2. lambda表达式格式:
- 中间:箭头操作符->
- 左边: lambda形参列表(其实就是接口中的抽象方法的形参列表)
- 右边:lambda体(其实就是重写的抽象方法的方法体)
3. lambda表达式的本质:

作为函数式接口的实例

  • 函数式接口:如果一个接口中,只声明了一个抽象方法,则此接口就称为函数式接口(我们可以在一个接口上使用@FunctionalInterface注解来检查一个接口是否为函数式接口)
4. lambda表达式使用(6种情况):
概述:
左边: lambda形参列表的参数类型可以省略(类型推断);如果lambda形参列表只有一个参数,
          其一对()也可省略
右边:lambda体应使用一对{}包裹;如果lambda体只有一条执行语句(可能是return语句),
           可以省略略这一对{}和return(省略大括号return必须也省略)

  • 语法格式一:无参无返回值
package Java8;

import org.junit.Test;

import java.util.Comparator;

/**
 * lambda表达式语法格式一: 无参无返回值
 */
public class LambdaTest1 {

    // 无参无返回值
    @Test
    public void test1() {
        Runnable r1 = new Runnable() {
            @Override
            public void run() {
                System.out.println("这是未使用Lambda表达式的");
            }
        };
        r1.run();
        System.out.println("------这是优美的分割线------");
        Runnable r2 = () -> System.out.println("这是使用了lambda表达式");
        r2.run();
    }

}

运行结果
在这里插入图片描述

  • 语法格式二:一个参数,无返回值
  • 语法格式三:类型推断,数据类型可以省略,因为可由编译器推断得出,称为“类型推断”
package Java8;

import org.junit.Test;

import java.util.function.Consumer;

/**
 * 语法格式二: 有一个参数,无返回值
 * 语法格式三: 类型推断
 */
public class LambdaTest2 {

    @Test
    public void test1() {
        Consumer<String> con1 = new Consumer<String>() {
            @Override
            public void accept(String s) {
                System.out.println(s);
            }
        };
        con1.accept("这是未使用lambda表达式");
        System.out.println("------优美的分割线------");
        Consumer<String> con2 = (String s) -> System.out.println(s);
        con2.accept("这是使用了lambda表达式");
        System.out.println("------优美的分割线------");
        // 类型推断,数据类型可以省略,因为可由编译器推断得出,称为“类型推断”
        Consumer<String> con3 = (s) -> System.out.println(s);
        con3.accept("类型推断");
        System.out.println("------优美的分割线------");
        Consumer<String> con4 = System.out::println;
        con4.accept("方法引用");
    }


}

运行结果
在这里插入图片描述

  • 语法格式四:只需要一个参数时,参数小括号可以省略
package Java8;

import org.junit.Test;

import java.util.function.Consumer;

/**
 * 语法格式四: 只需要一个参数时,参数的小括号可以省略
 */
public class LambdaTest3 {

    @Test
    public void test1() {
        Consumer<String> con1 = s -> System.out.println(s);
        con1.accept("只有一个参数时, 可省略小括号");
    }
}

运行结果
在这里插入图片描述

  • 语法格式五:两个及以上参数,多条执行语句,且有返回值
package Java8;

import org.junit.Test;

import java.util.Comparator;

/**
 * 语法格式五: 两个及以上参数, 多条执行语句,并且可有返回值
 */
public class LambdaTest4 {

    @Test
    public void test1() {
        Comparator<Integer> com1 = new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o1.compareTo(o2);
            }
        };
        int compare1 = com1.compare(12, 21);
        System.out.println("未使用lambda表达式:" + compare1);
        System.out.println("------优美的分隔符------");
        Comparator<Integer> com2 = (o1, o2) -> {
            System.out.println("语句1");
            System.out.println("语句2");
            return o1.compareTo(o2);
        };
        int compare2 = com2.compare(12, 21);
        System.out.println("使用lambda表达式:" + compare2);
    }
}

运行结果
在这里插入图片描述

  • 语法格式六:lambda体只有一条语句时,return与大括号都可省略
package Java8;

import org.junit.Test;

import java.util.Comparator;

/**
 * 语法格式六:lambda体只有一条语句时,return和大括号都可省略
 */
public class LambdaTest5 {

    @Test
    public void test() {
        Comparator<Integer> com1 = (o1, o2) -> {
            return o1.compareTo(o2);
        };
        System.out.println("未使用省略:" + com1.compare(12, 21));
        System.out.println("------优美的分隔符------");
        Comparator<Integer> com2 = (o1, o2) -> o1.compareTo(o2);
        System.out.println("使用省略:" + com2.compare(12, 21));
    }
}

在这里插入图片描述


2. 函数式接口

1. 什么是函数式接口?

只包含一个抽象方法的接口称为函数式接口(可使用@FunctionalInterface注解检验)

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

java.util.function包下定义了Java8丰富的函数式接口

函数式接口参数类型返回类型用途
Consumer< T > 消费型接口Tvoid对类型为T的对象应用操作,包含方法:void accept(T t)
Supplier< T >供给型接口T返回类型为T的对象,包含方法:T get()
Function<T, R>函数型接口TR对类型为T的对象应用操作,并返回结果。结果是R类型的对象。包含方法:R apply(T t)
Predicate< T >断定型接口Tboolean确定类型为T的对象是否满足某约束,并返回Boolean值。包含方法:boolean test(T t)
3. 代码示例
package Java8;

import org.junit.Test;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer;
import java.util.function.Predicate;

/**
 * 函数式接口
 */
public class LambdaTest6 {

    @Test
    public void test1() {
        happyTime(500, money -> System.out.println("去happy,花费: " + money));
    }

    public void happyTime(double money, Consumer<Double> con) {
        con.accept(money);
    }


    @Test
    public void test2() {
        List<String> list = Arrays.asList("东京", "西京", "南京", "北京", "天津", "上海");
        List<String> filterStrs = filterString(list, str -> str.contains("京"));
        System.out.println(filterStrs);
    }

    // 根据给定的规则,过滤集合中的字符串。此规则由Predicate的方法决定
    public List<String> filterString(List<String> list, Predicate<String> pre) {
        ArrayList<String> filterList = new ArrayList<>();
        for (String s : list) {
            if (pre.test(s)) {
                filterList.add(s);
            }
        }
        return filterList;
    }

}

运行结果
在这里插入图片描述


3. 方法引用

1. 概述
  • 当要传递给lambda体的操作已经有实现的方法了,可以使用方法引用
  • 方法引用可以看作是lambda表达式深层次的表达。换句话说,方法引用就是lambda表达式,也就是函数式接口的一个实例,通过方法的名字来指向一个方法,可以认为是lambda表达式的一个语法糖
2. 要求

实现接口的抽象方法的参数列表和返回值类型,必须和方法引用的方法的参数列表和返回值类型保持一致(适用于情况一、情况二)

3. 格式

使用操作符 "::"将类(或对象)与方法名分隔开来

4. 主要使用情况
  • 情况一:对象 :: 实例方法名
  • 情况二:类 :: 静态方法名
  • 情况三:类 :: 实例方法名
5. 代码示例
package Java8;

import org.junit.Test;

import java.util.Comparator;
import java.util.function.BiPredicate;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier;

public class MethodRefTest {

    /**
     * 情况一:对象::实例方法
     * Consumer中的void accept(T t)
     * printStream中的void println(T t)
     */
    @Test
    public void test1() {
        Consumer<String> con1 = str -> System.out.println("未使用方法引用:" + str);
        con1.accept("北京");
        System.out.println("------优美的分割线------");
        Consumer<String> con2 = System.out::println;
        con2.accept("使用方法引用");
    }

    // Supplier中的T get()
    // Employee中的String getName()
    @Test
    public void test2() {
        Employee employee = new Employee(1001, "zhangsan", 23, 5600);
        Supplier<String> supplier1 = () -> employee.getName();
        System.out.println(supplier1.get());
        System.out.println("------优美的分割线------");
        Supplier<String> supplier2 = employee::getName;
        System.out.println(supplier2.get());
    }

    // 情况二:类::静态方法
    // Comparator中的int compare(T t1, T t2)
    // Integer中的int compare(T t1, T t2)
    @Test
    public void test3() {
        Comparator<Integer> com1 = (t1, t2) -> Integer.compare(t1, t2);
        System.out.println(com1.compare(12, 21));
        System.out.println("------优美的分割线------");
        Comparator<Integer> com2 = Integer::compare;
        System.out.println(com2.compare(21, 12));
    }

    @Test
    public void test4() {
        Function<Double, Long> function1 = d -> Math.round(d);
        System.out.println(function1.apply(3.1));
        System.out.println("------优美的分割线------");
        Function<Double, Long> function2 = Math::round;
        System.out.println(function2.apply(3.5));
    }

    // 情况三: 类::实例方法
    // Comparator中的int compare(T t1, T t2)
    // String中的int t1.compareTo(t2)
    @Test
    public void test5() {
        Comparator<String> com1 = (s1, s2) -> s1.compareTo(s2);
        System.out.println(com1.compare("abc", "abd"));
        System.out.println("------优美的分割线------");
        Comparator<String> com2 = String::compareTo;
        System.out.println(com2.compare("abc", "abc"));
    }

    // BiPredicate中的boolean test(T t1, T t2);
    // String中的boolean t1.equals(t2)
    @Test
    public void test6() {
        BiPredicate<String, String> predicate1 = (s1, s2) -> s1.equals(s2);
        System.out.println(predicate1.test("abc", "abc"));
        System.out.println("------优美的分割线------");
        BiPredicate<String, String> predicate2 = String::equals;
        System.out.println(predicate2.test("abc", "abd"));
    }

    // Function中的R apply(T t)
    // Employee中的String getName()
    @Test
    public void test7() {
        Employee employee = new Employee(1001, "Jerry", 23, 6000);
        Function<Employee, String> function1 = e -> e.getName();
        System.out.println(function1.apply(employee));
        System.out.println("------优美的分隔符------");
        Function<Employee, String> function2 = Employee::getName;
        System.out.println(function2.apply(employee));
    }
}

运行结果

在这里插入图片描述


4. 构造器引用

1. 概述

构造器引用和方法引用类似,函数式接口的抽象方法的形参列表和构造器的形参列表一致。抽象方法的返回值类型即为构造器所属的类的类型

2. 代码示例
/**
     * 构造器引用
     * 和方法引用类似,函数式接口的抽象方法的参数列表和构造器的形参列表一致
     * 抽象方法的返回值类型即为构造器所属的类的类型
     */
    @Test
    public void test8() {
        Supplier<Employee> supplier1 = () -> new Employee();
        System.out.println(supplier1.get());
        System.out.println("------优美的分割线------");
        Supplier<Employee> supplier2 = Employee::new;
        System.out.println(supplier2.get());
    }

    // Function中的R apply(T t)
    @Test
    public void test9(){
        Function<Integer, Employee> function1 = id ->new Employee(id);
        Employee employee1 = function1.apply(1001);
        System.out.println(employee1);
        System.out.println("------优美的分割线------");
        Function<Integer, Employee> function2 = Employee::new;
        Employee employee2 = function2.apply(1002);
        System.out.println(employee2);
    }

运行结果

在这里插入图片描述

5. 数组引用

1. 概述

可以把数组看作是一个特殊的类,则写法与构造器引用一致

2. 代码示例
// 数组引用: 把数组看作是一个特殊的类,则写法就与构造器引用一致
    // Function中的R apply(T t)
    @Test
    public void test10() {
        Function<Integer, String[]> function1 = Length -> new String[Length];
        String[] arr1 = function1.apply(5);
        System.out.println(Arrays.toString(arr1));
        System.out.println("------优美的分割线------");
        Function<Integer, String[]> function2 = String[]::new;
        String[] arr2 = function2.apply(6);
        System.out.println(Arrays.toString(arr2));
    }

运行结果

在这里插入图片描述

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

SimpleZihao

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值