java8 lambda map<list>排序_Java 8 的这些特性,你知道吗?

作为一个心怀大教堂之愿景的搬砖码农,伴着一行代码一块砖的节奏,生产环境上 JDK 的版本从 1.4 逐步升级到 8。

而 JDK 都到 14 啦,而多数程序员编写的代码依然停滞在 Java 6,而每当看到小鲜肉写的代码,到处都是 Lambda 表达式、流式操作 Stream API ... ...,着实很懵 B。

为了能跟小鲜肉看齐,老码农很有必要熟练一下 Java 8 的那些新特性啦。

函数式接口

9632747f68880bf5e417895ba7b36a23.png

如 Runnable 源码所示,接口中定义了唯一一个抽象方法 run,那么类似这种的接口,在 Java 8 中称之为函数式接口,是 Java 8 引入的一个核心概念。

仔细看 Runnable 的源码,会发现引入了一个新的注解:@FunctionalInterface,用来表示这个接口是一个函数式接口。

@FunctionalInterface 这个注解是非必须的,若接口只包含一个方法的接口,虚拟机会自动判断,不过最好在接口上使用注解 @FunctionalInterface 进行声明。在接口中添加了 @FunctionalInterface 的接口,只允许有一个抽象方法,否则编译器也会报错。

Lambda 表达式

Lambda 表达式是推动 Java 8 发布的最重要新特性,可以使用 Lambda 实例化函数式接口。

在 Java 8 之前,匿名内部类,监听器和事件处理器的使用都显得很冗长,代码可读性很差,而采用 Lambda 表达式则可以使代码变的更加简洁紧凑。

781f4cf79adb1ec68fbab708d63d3804.png

(图片来源于网络)

代码片段一:

new Thread(new Runnable() {    @Override    public void run() {        System.out.println("我是一个孤独的线程");    }});

上面代码是创建线程的代码片段,这种代码很常见,而采用 Lambda 表达式,则会很简洁。

new Thread(() -> System.out.println("我是一个孤独的线程"));

很显然,代码确实是简化了不少。代码是否能读懂?没关系,稍后做解释。

代码片段二:

button.addActionListener(new ActionListener() {    @Override    public void actionPerformed(ActionEvent e) {        System.out.println("一个孤独的按钮被操作");    }});

上面的代码片段,是采用匿名内部类给按钮添加监听,而采用 Lambda 表达式,会是什么效果呢?

button.addActionListener((ActionEvent e) -> System.out.println("一个孤独的按钮被操作"));

哎呦,我去!妥妥的一行代码搞定,确实简化了不少。

烟味灭酒过半,是时候真正认识一下 Lambda 表达式啦。

Lambda 表达式,由三个部分组成。

第一部分:为一个括号内用逗号分隔的形式参数,参数是函数式接口里面方法的参数,例如代码片段二中的参数 ActionEvent e;

第二部分:为一个箭头符号:->;

第三部分:为方法体,可以是表达式和代码块。

具体语法如下:

(parameters) -> expression //当方法体为表达式时或(parameters) -> { statements; } //当方法体为代码块时

当方法体为表达式时,示例:

(int n1,int n2) -> return n1 + n2; //求和,该表达式的值作为返回值返回。

当方法体为代码块时,示例:

(String s) -> {System.out.println(s);} //打印,无返回值(int a, int b) -> { return a * b; }; //大括号中的返回语句

Lambda 表达式的语法很简单,不过有些准则还是要提一提。

  • 可选类型声明:不需要声明参数类型,编译器可以统一识别参数值。
  • 可选的参数圆括号:一个参数无需定义圆括号,但多个参数需要定义圆括号。
  • 可选的大括号:如果主体包含了一个语句,就不需要使用大括号。
  • 可选的返回关键字:如果主体只有一个表达式返回值则编译器会自动返回值,大括号需要指明表达式返回了一个数值。

那回头看看开篇提到的代码片段一、片段二,看完它们的简化过程,就很容易理解啦。

代码片段一,简化过程:

1c4f548e105de4b0f1af444824e9ce70.png

代码片段二,简化过程:

33ebdd1d08bf510266fbe84000045eb1.png

通过上面两段代码的简化过程,很显然使用 Lambda 表达式,可以很大程度上简化代码。

Stream:流式操作

Java 8 引入了流式操作(Stream),通过流式操作可以实现对集合的并行处理和函数式操作,想象成把要处理的元素集合看作一种流, 流在管道中传输, 并且可以在管道的节点上进行处理, 比如筛选, 排序,聚合等。

最重要的是:通过 Stream API 可以让程序员写出高效率、干净、简洁的代码。

Talk is cheap. Show me the code.

代码片段一:「filter」从集合中找出符合条件的数据。

List names = Arrays.asList("一猿小讲","一猿小将","一员小讲");// 统计名字列表中以 讲 结尾的个数long count = names.stream().filter(name -> name.endsWith("讲")).count();

代码片段二:「forEach」对集合中的数据进行迭代。

List names = Arrays.asList("一猿小讲","一猿小将","一员小讲");// 统计名字以 讲 结尾的的名字,并进行迭代输出names.stream().filter(name -> name.endsWith("讲")).forEach(name -> System.out.println(name));

代码片段三:「findAny」 vs「findFirst」。

List names = Arrays.asList("一猿小讲", "一猿小将", "一员小讲");// findAny 从列表中任意找出一个符合条件的数据Optional any = names.stream().filter(name -> name.endsWith("讲")).findAny();// findFirst 从列表中找出第一个符合条件的数据Optional first = names.stream().filter(name -> name.endsWith("讲")).findFirst();

代码片段四:「sorted」对流进行排序。

List names = Arrays.asList(" 8888 ", " 6666 ", " 9999 ");// 按照默认排序规则进行排序,当然可以自定义排序规则names.stream().sorted().forEach(name -> System.out.print(name));// 输出:6666  8888  9999

代码片段五:「distinct」去除重复元素。

List names = Arrays.asList("一猿小讲", "一猿小将", "一员小讲", "一猿小讲");// 去除重复元素,并迭代输出names.stream().distinct().forEach(name -> System.out.println(name));

代码片段六:「limit」获取指定数量的流。

List names = Arrays.asList("一猿小讲", "一猿小将", "一员小讲", "一猿小讲");// 获取 2 条数据,并迭代输出names.stream().limit(2).forEach(name -> System.out.println(name));

代码片段七:「map」映射每个元素对应的结果。

List names = Arrays.asList("一猿小讲", "一猿小将", "一员小讲", "一猿小讲");// 针对每个姓名进行追加 @NBnames.stream().map(name -> name + "@NB").forEach(name -> System.out.println(name));

还有很多 Stream 的 API,不再一一举例,用法套路都差不多,其实是一个熟能生巧的过程。

寄语写最后

本次,主要对 Java 8 中引入的部分新特性,进行初步的讲解。若想要快速投入实战,还要靠各位下去结合本文的代码片段进行多写、多悟,写着写着就有感觉啦,油从钱孔入而钱不湿唯手熟而已。

好了,本次就谈到这里,一起聊技术、谈业务、架构,少走弯路,不踩大坑。欢迎关注「一猿小讲」,会持续输出原创精彩分享,敬请期待!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值