Java8 Lambda 表达式

本文详细介绍了Java 8中的Lambda表达式概念及其语法格式,包括无参、单参、双参等不同情况下的使用方式。并通过Runnable接口、Callable接口的应用实例以及策略模式结合Lambda的案例展示了Lambda表达式的实用性和简洁性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

可参考博文:
http://blog.youkuaiyun.com/codejas/article/details/78810719
在学习Lambda 之前可以先了解一下java8 中关于接口的变化,以及函数式接口的定义及作用。

(一)Lambda 表达式介绍与语法格式

1.1Lambda 介绍

Lambda 是一个匿名函数,我们可以把 Lambda表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。这样做可以写出更简洁、更灵活的代码。

Lambda 表达式在Java 语言中引入了一个新的语法元素和操作符。这个操作符为 “->”, 该操作符被称为 Lambda 操作符或剪头操作符。它将 Lambda 分为两个部分。

左侧:指定了 Lambda 表达式需要的所有参数。
右侧:指定了 Lambda 体,即 Lambda 表达式要执行的功能。

1.2Lambda 格式

语法格式一:无参,无返回值

Runnable runnable = () -> System.out.println("Hello Lambda!");

语法格式二:Lambda 需要一个参数

 Consumer<String> fun1 = (parameter) -> System.out.println(parameter);

语法格式三:Lambda 只需要一个参数时,参数的小括号可以省略

 Consumer<String> fun2 = parameter -> System.out.println(parameter);

语法格式四:Lambda 需要两个参数,并且有返回值

BinaryOperator<Long> bo3 = (Long x, Long y) ->{
            System.out.println("实现函数接口方法!");
            return x + y;
        };

语法格式五:当 Lambda 体只有一条语句时,return 与大括号可以省略

BinaryOperator<Long> bo2 = (x, y) -> x + y;

1.3类型推断

BinaryOperator<Long> bo1 = (x, y) ->{
            System.out.println("实现函数接口方法!");
            return x + y;
        };

参考格式四,你会发现在传递参数的可以不用指定参数的类型,因为编译器推可以自动断出参数的数据类型。Lambda 表达式中无需指定类型,程序依然可以编译,这是因为 javac 根据程序的上下文,在后台推断出了参数类型。

(二)Lambda 表达式简单应用

2.1Runnable 无返回值应用

Runnable 接口中只有一个抽象方法run() ,并且使用了@FunctionalInterface 注解,表明它是一个函数式接口,所以它允许使用Lambda 表达式,下面是Java8 中Runnable 接口的源码。

@FunctionalInterface
public interface Runnable {

    public abstract void run();
}

以前我们编写多线程程序的时候一般都是定义一个类去实现Runnable 接口或者写一个Runnable 实现的匿名内部类来定义任务,然后启动它。现在我们了解了Lambda 表达式可以用更简单的方式去实现它。

public class LambdaTest {
    public static void main(String[] args) {

        /*  Runnable runnable = new Runnable() {
            @Override
            public void run() {
                System.out.println("Hello");      
            }
        };
        new Thread(runnable).start();*/

        // Lambda 方式定义实现
        Runnable runnable1 = () -> System.out.print("Hello");
        new Thread(runnable1).start();
    }
}

输出
Hello

2.2Callable 有返回值应用

Callable 接口与Runnable 接口类似,它也是只有一个抽象方法,只不过这个抽象方法是call(),具体源码如下。所以说它也是一个函数式接口。

@FunctionalInterface
public interface Callable<V> {

    V call() throws Exception;
}

Callable 接口允许我们从运行后的线程中获得返回值,下面我们用Lambda 表达式去实现一个简单的计算:开启一个线程去计算1-100 数字叠加的和。

public class LambdaTest {
    public static void main(String[] args) throws ExecutionException, InterruptedException {

        Callable<Integer> callable = () -> {
            int num = 0;
            for(int i = 0 ;i <= 100 ;i++)
                num += i;
            return num;
        } ;

        FutureTask<Integer> result = new FutureTask<>(callable);
        new Thread(result).start();
        System.out.print(result.get());
    }
}

输出
5050

2.3具体应用

下面是一个示例:定义一个员工类:包含字段username、age、gender和salary。这里我们以三种不同的方式区实现下面的需求。

需求:
①查找出薪水大于等于5000 的员工信息;
②查找出年龄大于等于20 岁的员工信息。

下面是员工类

public class Employee {
    private String username;
    private int age;
    private char gender; 
    private int salary;

    /** 省略get 与set 方法 */

    @Override
    public String toString() {
        return "Employee{" +
                "username='" + username + '\'' +
                ", age=" + age +
                ", gender=" + gender +
                ", salary=" + salary +
                '}';
    }
}

方式一:通过最简单的方式遍历集合,代码复用性很差

//所有员工信息的集合,下面的两种实现方式也都使用employeesList 集合
 List<Employee> employeesList = Arrays.asList(
            new Employee("小明",20,'男',5000),
            new Employee("小王",22,'男',4000),
            new Employee("小花",18,'女',5500),
            new Employee("小李",25,'男',6000));

 //通过遍历的形式将薪水条件符合的员工信息添加到集合中,返回条件符合的集合            
 public List<Employee> getEmployeesBySalary(List<Employee> employees) {
        List<Employee> list = new ArrayList<>();
        for(Employee employee : employees){
            if(employee.getSalary() >= 5000)
                list.add(employee);
        }
        return list;
    }
    @Test
    public void test1(){
        List<Employee> list = getEmployeesBySalary(employeesList);
        for(Employee employee : list){
            System.out.println(employee);
        }
    }

    //通过遍历的形式将年龄条件符合的员工信息添加到集合中,返回条件符合的集合
    public List<Employee> getEmployeesByAge(List<Employee> employees) {
        List<Employee> list = new ArrayList<>();
        for(Employee employee : employees){
            if(employee.getAge() >= 20)
                list.add(employee);
        }
        return list;
    }
    @Test
    public void test2(){
        List<Employee> list = getEmployeesByAge(employeesList);
        for(Employee employee : list){
            System.out.println(employee);
        }
    }

方式二:简单策略模式

策略接口


    //定义策略接口,因为该接口只有一个抽象方法所以也是一个函数式接口
    public interface MyStrategy<Employee>{
        boolean test(Employee employee);
    }

测试

    //返回符合条件的员工集合
     public List<Employee> filterEmployees(List<Employee> employees , MyStrategy<Employee> strategy) {
        List<Employee> list = new ArrayList<>();
        for(Employee employee : employees) {
            if(strategy.test(employee))    //根据策略添加符合对应条件的员工
                list.add(employee);
        }
        return list;
    }
    //为了避免复杂不定义策略类直接传入一个MyStrategy 形式的匿名内部类的实现,输出薪水条件符合的员工
    @Test
    public void test5(){
        List<Employee> list = filterEmployees(employeesList, new MyStrategy<Employee>() {
            @Override
            public boolean test(Employee employee) {
                return employee.getSalary() >= 5000;
            }
        });
        for(Employee employee : list){
            System.out.println(employee);
        }
    }

     //输出年龄条件符合的员工
    @Test
    public void test6(){
        List<Employee> list = filterEmployees(employeesList, new MyStrategy<Employee>() {
            @Override
            public boolean test(Employee employee) {
                return employee.getAge() >= 20;
            }
        });
        for(Employee employee : list){
            System.out.println(employee);
        }
    }

方式三:策略模式 + Lambda 表达式

    /** 这里仍然用到了策略模式,我们不用传入MyStrategy 的实现类,代码更加简洁了 */
    @Test
    public void test7(){
        List<Employee> list = filterEmployees(employeesList, (e) -> e.getSalary() >= 5000);
        for(Employee employee : list){
            System.out.println(employee);
        }
    }

    @Test
    public void test8(){
        List<Employee> list = filterEmployees(employeesList, (e) -> e.getAge() >= 20);
        for(Employee employee : list){
            System.out.println(employee);
        }
    }

(三)总结

这篇博文主要介绍了Lambda 的相关概念、使用方式,并通过具体的简单例子体会到了Lambda 的简洁。希望这篇博文能够为初学Lambda 的读者提供帮助。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值