gball个人知识库
首页
基础组件
基础知识
算法&设计模式
  • 操作手册
  • 数据库
  • 极客时间
  • 每日随笔
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
  • 画图工具 (opens new window)
关于
  • 网盘 (opens new window)
  • 分类
  • 标签
  • 归档
项目
GitHub (opens new window)

ggball

后端界的小学生
首页
基础组件
基础知识
算法&设计模式
  • 操作手册
  • 数据库
  • 极客时间
  • 每日随笔
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
  • 画图工具 (opens new window)
关于
  • 网盘 (opens new window)
  • 分类
  • 标签
  • 归档
项目
GitHub (opens new window)
  • 面试

  • 数据库

  • linux

  • node

  • tensorFlow

  • 基础组件

  • 基础知识

    • java集合

    • jvm调优

    • java并发编程

    • java网络编程

    • java8新特性

      • java8总结思维导图
      • lambda表达式详解
        • lambda表达式含义
        • 函数式接口
          • 1.什么是函数式接口
          • 2.自定义函数
          • 3.作为参数传递Lambda表达式
          • 4.四大内置核心函数式接口
        • 方法引用与构造器引用
          • 方法引用
          • 构造器引用(了解)
          • 数组引用(了解)
      • Stream API 详解
      • 记录JAVA8中Optional的使用
    • javaAgent

    • java高级

  • 算法与设计模式

  • 分布式

  • 疑难杂症

  • go学习之旅

  • 极客时间

  • 知识库
  • 基础知识
  • java8新特性
ggball
2022-01-10

lambda表达式详解

# lambda表达式含义

lambda表达式是对某些接口的简单实现,Lambda 规定接口中只能有一个需要被实现的方法,不是规定接口中只能有一个方法,这就是函数式接口。 ** lambda表达式引入“->” 左侧代表参数列表,右侧代表需要的功能,逻辑

Consumer<String> con = (c) -> System.out.println(c);
1

# 函数式接口

image.png

# 1.什么是函数式接口

(1)、只包含一个抽象方法的接口,称为函数式接口。 (2)、你可以通过Lambda表达式来创建该接口的对象。(若Lambda表达式抛出一个受检异常,那么该异常需要在目标接口的抽象方法上进行声明)。 (3)、我们可以在任意函数式接口上使用@FunctionalInterface注解,这样做可以检查它是否是一个函数式接口,同时javadoc也会包含一条声明,说明这个接口是一个函数式接口。

# 2.自定义函数

示例代码一:

@FunctionalInterface
public interface MyNumber{
    public double getValue();
}
1
2
3
4

示例代码二:函数式接口中使用泛型

@FunctionalInterface
public interface MyFunc<T>{
    public  T  getValue(T t);
}
1
2
3
4

# 3.作为参数传递Lambda表达式

示例代码:

public String strHandler(String str, MyFunction mf) {
        return mf.getValue(str);
    }
作为参数传递Lambda表达式:
String trimStr = strHandler("\t\t 你是大傻逼       ", (str) -> str.trim());
String upperStr = strHandler("abcdefg", (str) -> str.toUpperCase());
String newStr = strHandler("我大望江威武", (str) -> str.substring(2, 5));
1
2
3
4
5
6
7

作为参数传递Lambda表达式:为了将Lambda表达式作为参数传递,接收Lambda表达式的参数类型必须是与该Lambda表达式兼容的函数式接口的类型。

# 4.四大内置核心函数式接口

image.png

# 一、Consumer:消费型接口(void accept(T t))

**  **consumer 接口翻译过来就是消费者,顾名思义,该接口对应的方法类型为接收一个参数,没有返回值,可以通俗的理解成将这个参数'消费掉了',一般来说使用Consumer接口往往伴随着一些期望状态的改变或者事件的发生,例如最典型的forEach就是使用的Consumer接口,虽然没有任何的返回值,但是却向控制台输出了语句。 Consumer 使用accept对参数执行行为

/**
     * 消费型接口Consumer<T>
     */
    @Test
    public void test1 () {
        consumo(500, (x) -> System.out.println(x));
    }

    public void consumo (double money, Consumer<Double> c) {
        c.accept(money);
    }
1
2
3
4
5
6
7
8
9
10
11

以上为消费型接口,有参数,无返回值类型的接口。

# 二、Supplier:供给型接口(T get())

Supplier 接口翻译过来就是提供者,和上面的消费者相反,该接口对应的方法类型为不接受参数,但是提供一个返回值,通俗的理解为这种接口是无私的奉献者,不仅不要参数,还返回一个值,使用get()方法获得这个返回值

/**
     * 供给型接口,Supplier<T>
     */
    @Test
    public void test2 () {
        Random ran = new Random();
        List<Integer> list = supplier(10, () -> ran.nextInt(10));

        for (Integer i : list) {
            System.out.println(i);
        }
    }

    /**
     * 随机产生sum个数量得集合
     * @param sum 集合内元素个数
     * @param sup
     * @return
     */
    public List<Integer> supplier(int sum, Supplier<Integer> sup){
        List<Integer> list = new ArrayList<Integer>();
        for (int i = 0; i < sum; i++) {
            list.add(sup.get());
        }
        return list;
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

上面就是一个供给类型得接口,只有产出,没人输入,就是只有返回值,没有入参

# 三、Function<T, R>:函数型接口(R apply(T t))

function,顾名思义,函数的意思,这里的函数是指数学上的函数哦,你也可以说是严格函数语言中的函数,例如haskell里的,他接受一个参数,返回一个值,永远都是这样,是一个恒定的,状态不可改变的方法。其实想讲函数这个彻底将明白可以再开一篇博客了,所以这里不详细的说了。    上面说到,函数接口是对行为的抽象,因此我方便大家理解,就用java中的方法作例子。

/**
     * 函数型接口:Function<R, T>
     */
    @Test
    public void test3 () {
        String s = strOperar(" asdf ", x -> x.substring(0, 2));
        System.out.println(s);
        String s1 = strOperar(" asdf ", x -> x.trim());
        System.out.println(s1);
    }

    /**
     * 字符串操作
     * @param str 需要处理得字符串
     * @param fun Function接口
     * @return 处理之后得字符传
     */
    public String strOperar(String str, Function<String, String> fun) {
        return fun.apply(str);
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

上面就是一个函数型接口,输入一个类型得参数,输出一个类型得参数,当然两种类型可以一致。

# 四、Predicate:断言型接口(boolean test(T t))

predicate<T,Boolean> 谓语接口,顾名思义,中文中的‘是’与‘不是’是中文语法的谓语,同样的该接口对应的方法为接收一个参数,返回一个Boolean类型值,多用于判断与过滤,当然你可以把他理解成特殊的Funcation<T,R>,但是为了便于区分语义,还是单独的划了一个接口,使用test()方法执行这段行为

/**
     * 断言型接口:Predicate<T>
     */
    @Test
    public void test4 () {
        List<Integer> l = new ArrayList<>();
        l.add(102);
        l.add(172);
        l.add(13);
        l.add(82);
        l.add(109);
        List<Integer> list = filterInt(l, x -> (x > 100));
        for (Integer integer : list) {
            System.out.println(integer);
        }
    }

    /**
     * 过滤集合
     * @param list
     * @param pre
     * @return
     */
    public List<Integer> filterInt(List<Integer> list, Predicate<Integer> pre){
        List<Integer> l = new ArrayList<>();
        for (Integer integer : list) {
            if (pre.test(integer))
                l.add(integer);
        }
        return l;
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31

上面就是一个断言型接口,输入一个参数,输出一个boolean类型得返回值。

# 总结

四大函数接口和他们引申出的各类接口,终点是对不同种类行为的封装导致了设计出不同的函数接口,另外在使用函数接口或者lambda表达式的时候,要注意lambda对值封闭这个特性。

# 方法引用与构造器引用

# 方法引用

当要传递给Lambda体的操作,已经有实现的方法,就可以使用方法引用! 实现抽象方法的参数列表,必须与方法引用方法的参数列表保持一致 方法引用使用操作符“::”将方法名和对象或者类的名字分隔开来。 三种使用情况: 对象::实例方法 类::静态方法 类::实例方法

例如

BinaryOperator bo=(x,y)->Math.pow(x,y)

等同于:

BinaryOperator<Double> bo=Math::pow;

 /*
    * 类::静态方法
    * Comparator的Compare(T t1,T t2)
    * Integer的Compare(T t1,T t2)
    * */
Comparator<Integer> com1 = (t1,t2) -> Integer.compare(t1,t2);
        System.out.println(com1.compare(21,20));
        System.out.println("------------");
        Comparator<Integer> com2 = Integer::compare;
        System.out.println(com2.compare(12,14));
1
2
3
4
5
6
7
8
9
10

# 构造器引用(了解)

格式:ClassName::new 与函数式接口相结合,自动与函数式接口中方法兼容。 可以把构造器引用赋值给定义的方法,与构造器参数列表要与接口中抽象方法的参数列表一直 例如:

Function<Integer,MyClass> fun=(n)->new MyClass(n);

等同于:

Function<Integer,MyClass> fun=MyClass::new;

# 数组引用(了解)

格式:type[]::new 例如:

Function<Integer,Integer[]> fun=(n)->new Integer[n];

等同于:

Function<Integer,Integer[]> fun=Integer[]::new;

上次更新: 2025/06/04, 15:06:15
java8总结思维导图
Stream API 详解

← java8总结思维导图 Stream API 详解→

最近更新
01
AIIDE
03-07
02
githubActionCICD实战
03-07
03
windows安装Deep-Live-Cam教程
08-11
更多文章>
Theme by Vdoing
总访问量 次 | 总访客数 人
| Copyright © 2021-2025 ggball | 赣ICP备2021008769号-1
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式
×

评论

  • 评论 ssss
  • 回复
  • 评论 ssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssss
  • 回复
  • 评论 ssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssss
  • 回复
×