Java - Lambda表达式、函数式接口

本文介绍Java 8中引入的Lambda表达式及其简化规则,对比匿名内部类,并讲解函数式接口的概念及常见函数式接口如Supplier、Consumer、Function和Predicate的使用方法。

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

第一章 Lambda表达式

1. Lambda表达式概述

  • JDK 8 新特性
  • 作用:简化函数式接口的匿名内部类的写法
  • 标准格式:
    (参数类型 参数名称) -> { 方法体 }
    

2. 接口实现类、匿名内部类、Lambda表达式对比

package com.note.functions_interface_test;
public class Test {
    public static void main(String[] args) {
        //如何使用接口中的方法?

        //方式一 普通形式
        //1.创建接口的实现类MyInterfaceImpl,实现接口MyInterface,覆盖重写抽象方法
        //2.创建实现类对象,调用方法
        new MyInterfaceImpl().method(10);

        //方式二 匿名内部类
        //不需要自己创建实现类了,从out中的.class文件可以看出,jvm会帮我们创建接口实现类,
        new MyInterface(){
            @Override
            public void method(int a) {
                System.out.println(a);
            }
        }.method(10);

        //方式三 Lambda简化匿名内部类
        ((MyInterface)(int a)-> System.out.println(a)).method(10);
    }
}
//接口
@FunctionalInterface
interface MyInterface {
    void method(int a);
}
//接口的实现类
class MyInterfaceImpl implements MyInterface{
    @Override
    public void method(int a){
        System.out.println(a);
    }
}
package com.alibaba.lambda_practice;

/*
通过创建线程,对比lambda表达式的简便之处。
 */
public class Demo01 {

    public static void main(String[] args) {
        //方式一:老老实实创建Runnable接口子类
        //1.新建Runable接口的子类RSON,创建子类对象
        RSON rson = new RSON();
        //2.创建Thread类对象,构造方法传入子类对象rson
        Thread t1 = new Thread(rson);
        //3.开启线程
        t1.start();

        //方式二:用匿名内部类来创建Runnable接口实现类对象,省去单独建一个类了
        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("方式二:简单打印一下");
                for (int i = 0; i < 10; i++) {
                    System.out.print(i);
                }
                System.out.println("\n==================");
            }
        });
        t2.start();

        //方式三: Runnable符合函数式接口,用lambda表达式简化
        Thread t3 = new Thread(() -> {
            System.out.println("方式三:简单打印一下");
            for (int i = 0; i < 10; i++) {
                System.out.print(i);
            }
            System.out.println("\n==================");
        });
        t3.start();
    }

}

class RSON implements Runnable {

    @Override
    public void run() {
        System.out.println("方式一:简单打印一下");
        for (int i = 0; i < 10; i++) {
            System.out.print(i);
        }
        System.out.println("\n==================");
    }
}

3. Lambda表达式的简化规则

  • 如果参数只有一个,可以省略参数类型 和 小括号
  • 如果方法体只有一行代码,可以省略{},同时也要省略分号;
  • 如果方法体只有return 一行代码,可以省略{},同时也要省略分号; return
package com.note.functions_interface_test;
public class Test02 {
    public static void main(String[] args) {
        //接口的抽象方法无参数
        ((MineInterface) () -> 
                System.out.println("hello")
        ).method();

        //接口的抽象方法 有一个参数,可省略形参类型,形参小括号
        ((MineInterface) (int a) ->
                System.out.println("hello")
        ).method(10);
        ((MineInterface) (a) ->
                System.out.println("hello")
        ).method(10);
        ((MineInterface) a ->
                System.out.println("hello")
        ).method(10);

        //接口的抽象方法 有多个参数
        ((MineInterface) (int a, int b) -> {
            System.out.println("hello");
        }).method(10, 20);

        //方法体只有一条语句,可省略{ } 和 语句后的;
        ((MineInterface) (int a, int b) -> System.out.println("hello")).method(10, 20);
        
        //方法体只有一条return语句,可省略 {} return ;
        ((MineInterface) (int a, int b) -> 0).method(10, 20);
    }
}
//接口
@FunctionalInterface
interface MineInterface {
    void method();
    void method(int a);
    void method(int a, int b);
    int method(int a, int b);
}

4. Lambda注意

  • 用匿名内部类时有时会爆这样的编译异常Variable used in lambda expression should be final or effectively final
  • 原因: Lambda表达式中使用的变量应该是 final 或者有效的 final修饰的。

https://www.runoob.com/

package com.alibaba.lambda_practice;

/*
异常提示:Variable used in lambda expression should be final or effectively final
*/
public class Demo02 {
    public static void main(String[] args) {
        int n = 10;

        new Thread(new Runnable() {
            @Override
            public void run() {
                n = 20;
                System.out.println(n);
            }
        }).start();
    }
}


5. 匿名内部类与Lambda表达式的区别

  1. 匿名内部类可以用于接口、抽象类。Lambda表达式只能用于函数式接口
  2. 匿名内部类会自动生成一个接口的实现类.class文件,Lambda表达式Lambda的字节码文件会动态生成

第二章 函数式接口

1. 函数式编程思想

  • 面向对象过分强调“必须通过对象的形式来做事情”。找对象,通过对象调用方法。
  • 而函数式思想则尽量忽略面向对象的复杂语法——强调做什么,而不是以什么形式做

2. 函数式接口

  • 在Java中是指:有且仅有一个抽象方法的接口
  • 接口可加注解@FunctionalInterface //加上注解,编译器会强制检查该接口是否符合函数式接口
package com.note.functions_interface_test;
@FunctionalInterface 
interface MyFunctionalInterface{
    void method();
}

3. 常用函数式接口

  • JDK提供了大量常用的函数式接口以丰富Lambda的典型使用场景,它们主要在java.util.function包中被提供。

3.1 Supplier接口

  • java.util.function.Supplier<T>接口,“供应商”
  • 抽象方法T get()用来获取一个泛型参数指定类型的对象数据
  • 练习题:
package com.note.functions_interface_test.supplier_test;

import java.util.Arrays;
import java.util.function.Supplier;

public class Test {
    public static void main(String[] args) {
        //Supplier练习题求数组最大值
        int[] nums = {20, 3, 4, 5, 6};
        int m = getMethod(nums, () -> {
            int max = nums[0];
            for (int num : nums) {
                if(max < num){
                    max = num;
                }
            }
            return max;
        });
        System.out.println(m);
        
        //Supplier练习题求数组元素之和
        int s = getMethod(nums,()->{
			int sum = 0;
			for(int i = 0; i < nums.length ; ++i){
				sum += nums[i];
			}
			return sum;
		});
        System.out.println(s);
        
        //这样写的好处,将方法内具体怎么操作交给调用者决定
    }
    public static int getMethod(int[] nums, Supplier<Integer> supplier) {
        return supplier.get();
    }
}

3.2 Consumer接口

  • java.util.function.Consumer<T>接口 消费一个数据
  • 抽象方法void accept(T t),意为消费一个指定泛型的对象数据。
  • 练习题: 给你一个字符串,请按照大写的方式进行消费
package com.note.functions_interface_test.consumer_test;

import java.util.function.Consumer;

public class Test {
    public static void main(String[] args) {
        method("abcd", str -> System.out.println(str.toUpperCase()));
    }

    public static void method(String str, Consumer<String> consumer) {
        consumer.accept(str);
    }
}

3.3 Function接口

  • java.util.function.Function<T,R> 接口用来根据一个类型的数据得到另一个类型的数据,前者称为前置条件,后者称为后置条件。
  • 抽象方法 R apply(T t),根据类型T的参数获取类型R的结果。使用的场景例如:将String类型转换为Integer类型。
  • 练习题:
package com.note.functions_interface_test.function_test;

import java.util.function.Function;

public class Test {
    public static void main(String[] args) {
        //将String类型的"hello" 转为Integer
        method("100",t->Integer.parseInt(t));
        method("100",Integer::parseInt); //方法引用 简化
    }

    public static <T> void method(T t,Function<T,Integer> function){
        function.apply(t);
    }
}

3.4 Predicate接口

  • java.util.function.Predicate<T>接口 有时候我们需要对某种类型的数据进行判断,从而得到一个boolean值结果。
  • 抽象方法:boolean test(T t) 用于条件判断的场景
  • 练习题:
package com.note.functions_interface_test.predicate_test;

import java.util.function.Predicate;

public class Test {
    public static void main(String[] args) {
        //1.练习:判断字符串长度是否大于5
        System.out.println(method("abcde", (String str) -> str.length() > 5));
        System.out.println(method("abcdesf", str -> str.length() > 5));
        //2.练习:判断字符串是否包含"H"
        System.out.println(method("ABCDH", (String str) -> str.contains("H")));
        System.out.println(method("ABCDH", str -> str.contains("H")));
    }

    public static <T> boolean method(T t, Predicate<T> predicate) {
        return predicate.test(t);
    }
}

第三章 参考资料

B站黑马视频

Lambda表达式与匿名内部类的区别

匿名内部类中变量需要final修饰

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值