Lambda表达式详细总结,理论加案例

满怀忧思,不如先干再说!通过学习,重新定义自己!

JDK版本迭代

近几年JDK更新速度非常快,2022年9月20号发布JDK19,作为Java语言的使用者,尤其是新入行和想要入行的朋友,不要被这么快的更新速度吓到,更不要被贩卖了焦虑,因为多数版本仅为过渡,如JDK19在2023年3月将会被JDK20替代,完全不必慌张,近几年JDK更新如此频繁,真正多的变化其实还在于JDK8之中。本文章为系列文章,后续内容持续推出!

从1996年1月JDK1.0正式发布到现在,目前有三个主流的长期支持【LTS】版本为JDK8、JDK11、JDK17,因为JDK11并没有特别大的优化更新,46.45%的企业还是使用2014年3月14号发布的JDK8,JDK17使用率仅在0.37%

JDK8重要更新

继《JDK8更新——官网原版详解》之后,本篇主要介绍一下JDK更新在编码层面的重要变化,也就是我们一线程序员需要掌握的都有哪些,网络上有许多教程,但恰恰是教程太多五花八门,不知所云,一个新特性怼六七个小时实属浪费时间,学无涯生也有涯,很多玩意我们知道就行不需要去掌握,不要去浪费时间,结合实际工作,我总结了如下几点:

小贴士:本篇文章主要讲解Lambda表达式

Lambda表达式

Lambda表达式支持将一个方法【行为】当做参数传递,这种编程方式称为【函数式编程】,这种编程方式最大的特点就是代码紧凑,减少冗余代码,让编程边的更加简洁,而Java最大的弊端就在于代码臃肿,在Python,Scala、JavaScript等语言中也都引入函数式编程,而Java通过Lambda表达式实现函数式编程势在必行!

语法格式

Lembda表达式通过左侧的参数,右侧的表达式和中间的右箭头组成:

// 有参有返回值
(parameter1,parameter12,...) -> {
	expression;
	...
	return xxx;
}
// 无参有返回值 
() -> {
	expression;
	...
	return xxx;
}

// 有参无返回值
(parameter1,parameter12,...) -> {
	expression;
	...
}

// 无参无返回值
() -> {
	expression;
	...
}

实现原理

重点掌握

无参实现

Lambda表达式实现需要依赖于函数式接口,JDK内置了一些函数式接口,使用Supplier来实现无参Lambda讲解,至于函数式接口怎么自定义在后续单独介绍,保证连贯性,在此不穿插函数式接口其他内容!

Supplier接口为JDK内置的供给型接口,特点为无参数但是有返回值,定义如下:

代码:

package com.stt;

import java.util.function.Supplier;

/**
 * 无参函数式接口
 */
public class NoArgsMain {
    public static void main(String[] args) {
        // 1、原始方式实现
        Supplier s1 = new Supplier() {
            @Override
            public Integer get() {
                return 1024;
            }
        };
        // 通过get方法获取返回值
        System.out.println(s1.get());
        
        // 2、通过Lambda表达是实现
        Supplier s2 = () -> {
            return "Lambda实现";
        };
        // 通过get方法获取返回值
        System.out.println(s2.get());
    }
}

简化代码:

如果Lambda表达式有返回值且代码体只有一行代码时,return和大括号可以省略不写

// 1、简化后代码如下
public class NoArgsMain {
    public static void main(String[] args) {
        
        /* 
        1、函数体只有一行代码,return可以省略
        2、只有一行代码大括号可以省略
        */
        Supplier s2 = () -> "Lambda实现";
        // 通过get方法获取返回值
        System.out.println(s2.get());
    }
}

// 2、如果有多行,则不可以省略
public class NoArgsMain {
    public static void main(String[] args) {
        // 通过Lambda表达是实现,因为函数体有其他代码,不可省略return和大括号
        Supplier s2 = () -> {
            System.out.println("我是Lambda表达式体");
            return "Lambda实现";
        };
        // 通过get方法获取返回值
        System.out.println(s2.get());
    }
}

有参实现

通过JDK内置Consumer接口实现,接收参数但没有返回值,定义如下:

代码实现:

package com.stt;

import java.util.function.Consumer;

public class HasArgsMain {
    public static void main(String[] args) {
        // 1、原始实现
        Consumer consumer1 = new Consumer() {
            @Override
            public void accept(Integer param) {
                // 没有返回值
                System.out.println("我是消费型接口,只进不出哦!" + param);
            }
        };
        // 调用accpet方法消费数据
        consumer1.accept(1024);
        
        // 2、Lambda写法,
        Consumer consumer2 = (str) -> {
            System.out.println("我是Lambda消费型接口," + str);
        };
        consumer2.accept("石添添");
    }
}

简化写法:

public class HasArgsMain {
    public static void main(String[] args) {
        /*
        1、如果只有一个参数则可以省略小括号
        2、代码体只有一行代码则可以省略大括号
        */
        Consumer consumer2 = str -> System.out.println("我是Lambda消费型接口," + str);
        consumer2.accept("石添添");
    }
}

总结:

有没有那么一点点感觉,Lambda表达式简化了代码,让编码更加简洁,接下来我们通过更多案例对比进一步理解和使用Lambda表达式

Lambda实现线程创建

通过Runnable接口创建线程,如果想使用Lambda那么Runnable应该是一个函数式接口,函数式接口的特点是只有一个抽象方法,Runnable接口定义如下:

代码实现:

public class ThreadMain {
    public static void main(String[] args) {
        // 1、原始方式创建线程
        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName() + "线程启动");
            }
        }).start();

        // 2、使用Lambda表达式简写方式创建线程
        new Thread(() -> System.out.println(Thread.currentThread().getName() + "线程启动")).start();
    }
}

Lambda实现集合遍历

集合操作新增forEach方法,接收一个Consumer类型对象,上边【有参实现】中介绍了,它是一个函数式接口

代码实现:

public class CollectionMain {
    public static void main(String[] args) {
        // 创建集合
        List list = new ArrayList<>();
        // 添加数据
        list.add("艾斯!");
        list.add("赛罗!");
        list.add("杰克!");
        list.add("雷欧!");
        // 1、原始遍历方式
        list.forEach(new Consumer() {
            @Override
            public void accept(String s) {
                System.out.println(s);
            }
        });
        // 2、因为Consumer是一个函数式接口,可以使用Lambad
        // 思考:简写形式怎么实现呢?
        list.forEach((str) -> {
            System.out.println(str);
        });
    }
}

Lambda实现集合排序

集合排序可以使用sort方法,sort方法接收一个Comparator类型数据

Comparator接口定义

小贴士:该接口中还有几个默认实现方法和静态方法,因为只有一个抽象方法所以也是函数式接口

代码实现:

public class CollectionSortMain {
    public static void main(String[] args) {
        List personList = new ArrayList<>();
        personList.add(new Person("李小白",24));
        personList.add(new Person("张二三",21));
        personList.add(new Person("王五六",30));
        // 遍历
        System.out.println("排序前-------------------");
        personList.forEach(person -> System.out.println(person));
        // 排序,参数1 - 参数2为升序排序
        System.out.println("排序后-------------------");
        personList.sort(new Comparator() {
            @Override
            public int compare(Person p1, Person p2) {
                return p1.getAge() - p2.getAge();
            }
        });
        personList.forEach(person -> System.out.println(person));
        // Lambda排序,参数2 - 参数1为降序排序
        System.out.println("Lambda排序-------------------");
        // 直接简写形式,只有一行代码,大括号和return可以省略
        personList.sort((p1,p2) -> p2.getAge() - p1.getAge());
        personList.forEach(person -> System.out.println(person));
    }
}

总结

后续将陆续推出其它新特性,记得关注哦!

展开阅读全文

页面更新:2024-04-29

标签:表达式   简写   括号   线程   函数   接口   参数   案例   理论   类型   代码   方法   详细

1 2 3 4 5

上滑加载更多 ↓
推荐阅读:
友情链接:
更多:

本站资料均由网友自行发布提供,仅用于学习交流。如有版权问题,请与我联系,QQ:4156828  

© CopyRight 2008-2024 All Rights Reserved. Powered By bs178.com 闽ICP备11008920号-3
闽公网安备35020302034844号

Top