草庐IT

一文带你攻克JDK新特性

叶秋学长 2023-07-12 原文

1.Java8 新特性介绍

1.1 实验介绍

在国内,Java8 是当之无愧的普及率最高的 JDK 版本,从笔者工作开始, 就一直使用的是 JDK8 的版本,虽然现在 JDK19 即将面世,但是似乎依旧无法 动摇 JDK8 在国内的地位。这里面最主要的原因就是 JDK8 足够稳定,功能足够 优秀,而替换 JDK 版本会有潜在的风险。既然 JDK8 如此稳定,就意味着需要 很好的掌握 JDK8 中的特性,本次实验就会对 JDK8 做一个详细的介绍,并通过 一些简单例子做个基础的入门。

知识点

1. JDK 的版本化

2. Lambda 表达式

3. 函数式接口

4. 方法引用与构造器引用

5. Stream 表达式

6.接口中的默认方法和静态方法

7. Optional

8. 新的时间日期 API

1.2 JDK 的版本化

在正式开始本次实验之前,有个概念需要了解。JDK 在发展过程中,分为 LTS 版 本和 non-LTS 版本,其中 LTS 版本表示这是一个长期支持的版本,而 non-LTS 表示这是一个不会长期支持的版本。

从上面这张图可以看到,目前的 JDK 版本中,8、11、17、21 会是被长期支持 的版本,并且对 JDK8 的支持时间达到了 2030 年,比 JDK11 和 JDK 17 还要 长,因此学好 JDK8 更加重要了。

 1.3Lambda 表达式

Lambda 表达式是 JDK8 中最大的一次更新,Lambda 是一个匿名函数,它允许你 通过表达式来代替功能接口,使用 Lambda 可以让代码变得更加简洁。简单来讲, Lambda 使用更简洁的方式给接口添加实现方法。 比如下面这段代码,以前是这样写的,Runnable 后的匿名类需要被完整的写出 来。

public class AnonymousDemo {

public static void main(String[] args) { Runnable runnable = new Runnable() {

@Override

public void run() { System.out.println("do something"); } }; Thread thread = new Thread(runnable); thread.start(); } }

当 Lambda 出现后,这种写法就变成了下面这样

public class AnonymousDemo {

public static void main(String[] args) {

new Thread(() -> System.out.println("do something")).start(); } }

是不是一下子变简单了?

1.4 函数式接口

函数式接口就是那些只有一个抽象方法的接口,JDK8 中内置了四种函数式接口:

Consumer : void accept(T t); Supplier : T get(); Function : R apply(T t); Predicate : boolean test(T t)

这四种内置的函数式接口提供了四种不同类型的代码写法,以 Consumer 为例。 Consumer 是一种消费型接口,提供了一个参数、无返回值的接口方法,就和消 费一样,花出去就没有了。

import java.util.function.Consumer;

public class ConsumerTest {

public static void main(String[] args) { consume(100,money -> { System.out.println("消费了"+money+"元"); }); }

public static void consume(double money, Consumer consumer){ consumer.accept(money); } }

在上面的例子中,consume 方法有两个入参,一个 double 类型的金额和一 个 Consumer 函数接口,方法的实现就是执行 Consumer 函数接口默认的 accept 方法。在使用 consume 方法中,就可以直接通过 Lambda 表达式来实现 函数式接口的调用,让代码更加简洁。

1.5 方法引用与构造器引用

方法引用使得开发者可以直接引用现存的方法、Java 类的构造方法或者实例对 象。构造器引用和方法引用类似,主要用途往往是创建对象。方法引用和构造器 引用主要是配合 Lambda 表达式,使得表达式变得更加简单。 比如要输出一个 List 中的数据,使用方法引用的话通 过 System.out::println 就实现了。

import java.util.Arrays;import java.util.List;

public class Test2 {

public static void main(String[] args) { List list = Arrays.asList("a","b"); list.forEach(System.out::println); } }

1.6 stream 表达式

JDK8 中两个最重大的改变,一个是 Lambda 表达式,另外一个就是 Stream API。Stream 是 JDK8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的 操作,可以执行非常复杂的查找、过滤和映射数据等操作。 使用 Stream API 对集合数据进行操作,就类似于使用 SQL 执行的数据库查询。 也可以使用 Stream API 来并行执行操作。简而言之,Stream API 提供了一种 高效且易于使用的处理数据的方式。 比如我要获取 List 中大于 50 的数据,就可以这样写

import java.util.Arrays;import java.util.List;import java.util.stream.Co llectors;public class StreamTest {

public static void main(String[] args) { List list = Arrays.asList(1,2,3,51,52,53); List filterList = list.stream().filter(x -> x>50).collec t(Collectors.toList()); fliterList.forEach(System.out::println); } }

上面这段代码将大于 50 的数据传到了 filterList 中并输出。大家可以打开右 边的实验环境验证最终的输出结果。

1.7 接口中的默认方法和静态方法

在 JDK8 之前,接口中的方法只能声明无法实现,在 JDK8 中,接口中可以添加 默认方法和静态方法了。 首先介绍默认方法,通过 default 修饰的方法可以在接口中增加实现。

public interface MyInterface {

default void defaultMethod(){ System.out.println("hello"); } }

另外是接口中的静态方法:

public interface MyInterface {

public static void staticMethod(){ System.out.println("hello"); } }

更多的介绍会在后面的实验中给出。

1.8 Optional

Optional 类是 JDK8 的新特性,是一个可以为 null 的容器对象,Optional 往 往被用来做空值的判断。 集合类型的判空在某些场景下会十分啰嗦,比如当你接收到的 Map 是这样的

{"user":{"info":{"address":"hz"}}}

这种时候如果按照常规的写法,需要写多层 if 语句进行判空

if (map.get("user")!=null){ Map user = (Map) map.get("user");

if (user.get("info")!=null){ Map info = (Map) user.get("info");

if (info.get("address")!=null){ String address = (String) info.get("address"); System.out.println(address); } } }

if 里面套着 if,结构十分复杂,这个时候我们就可以使用 Optional

String address=Optional.ofNullable(map) .map(m->(Map)m.get("user")) .map(user->(Map)user.get("info")) .map(info->(String)info.get("address")) .orElse(null);

一下子就变得简洁明了。

1.9 新的时间日期 API

时间类一直是代码开发中经常用到的东西,时间类更新到 JDK8 版本期间,一共 迭代了三次。分别是 Date 类、Calendar 类和 LocalDateTime 类。 这一次推出来的 LocalDateTime 类,终于让 Java 中的时间类变得很好用了

import java.time.Instant;import java.time.LocalDate;import java.time.Loc alDateTime;import java.time.LocalTime;

public class DateTest {

public static void main(String[] args) { Instant now = Instant.now();

//获取纳秒级别的时间戳

System.out.println(now.toEpochMilli());LocalDate localDate=LocalDate.now(); LocalTime localTime=LocalTime.now(); LocalDateTime ldt = LocalDateTime.now(); System.out.println(localDate); System.out.println(localTime); System.out.println(ldt); } }

大家能猜出上面的这段代码会返回怎样的数据吗?

1.10 实验总结

本次实验对 JDK8 中的一些重要特性做了简单的介绍,可以看到这些新特性让 Java 变得更加有趣了。在接下来的实验中,针对上述的内容会有更详细的介绍。

2.学透 Lambda 表达式

2.1 实验介绍

Java8 是一个跨时代的更新,在这个版本中,出现了很多新特性,其中 Lambda 表 达式就是 Java8 中十分重要的新特性。本次实验将基于 Java8 详细介绍 Lambda 表达式的使用,简化代码的编写。

知识点

1. 匿名类

2. 函数式接口

3. Lambda 表达式介绍

4.Lambda 表达式的变量作用域

5.Lambda 表达式在实际中的应用

2.1 匿名类

在正式介绍 Lambda 表达式之前,首先需要知道的一个知识点是匿名类。 因为 Java 中的 Lambda 表达式最重要的用法就是简化匿名类的使用。匿名类 可以同时声明和实例化一个类,匿名类和本地类十分相似,只是匿名类没有名称。 比如下面有一个叫 Person 的接口,里面有一个 sayHello 方法

public interface Person {

void sayHello(); }

接着创建一个类 Son 实现 Person 接口,实现里面的 sayHello 方法。 在不用匿名类的情况下,就需要新建一个类实现 Person,然后重写 sayHello 方 法,就像下面这样

public class Son implements Person{

@Override

public void sayHello() { System.out.println("son say hello"); } }

如果这个 Son 类只是用来在某个类中执行特定任务的话,就可以使用更加简洁 的匿名内部类写法,而无需创建这个 Son 类

public class AnonymousDemo {

public static void main(String[] args) { Person person = new Person(){

@Override

public void sayHello() { System.out.println("anonymous say hello"); } }; person.sayHello(); } }

在上面这个例子中,虽然没有创建实体的类,但是通过一个匿名类的方式实现了 本地类的效果。总结下来,匿名类是不能有名字的类,它们不能被引用,只能在 创建时用 new 语句来声明它们。

2.2 函数式接口

在正式学习 Lambda 表达式之前,还有一个很重要的概念需要了解,那就 是函数式接口。函数式接口主要就是给 Lambda 表达式使用的。 函数式接口可以使用 @FunctionalInterface 来修饰,如果一个接口是函数式接 口,那么这个接口就可以使用 Lambda 表达式来写。 函数式接口有下面几个特征:

. 只能有一个抽象方法。

. 允许定义默认方法

. 允许定义静态方法

. 允许定义 java.lang.Object 里的 public 方法

拿 java.util 包下的 Comparator 接口举个例子,下面是这个接口的一部分代码

@FunctionalInterfacepublic interface Comparator {

int compare(T o1, T o2);

boolean equals(Object obj);

default Comparator reversed() {

return Collections.reverseOrder(this); }

public static super T>> Comparator reverse Order() {

return Collections.reverseOrder(); } }

从代码中可以看到,Comparator 完美符合了函数式接口的特性,只能有一个抽 象方法 compare,有默认方法,有静态方法,有 java.lang.Object 里的 public 方 法。

2.3 Lambda 表达式

匿名类的一个问题是,如果匿名类的实现非常简单,例如只包含一个方法的接口, 那么匿名类的语法可能看起来笨拙且不清楚。因此 Lambda 表达式出现了,

Lambda 允许通过表达式来代替功能接口,使用 Lambda 可以让代码变得更加 简洁。

Lambda 只能在函数式接口中使用,上面这段使用匿名类的方法,换成 Lambda

表达式之后就变成了下面这样:

public class AnonymousDemo {

public static void main(String[] args) { Person person = () -> System.out.println("anonymous say hello"); person.sayHello(); } }

一下子变得更加简单了。

Lambda 表达式的语法格式如下所示:

(parameters) -> expression

(parameters) ->{ statements; }

lambda 表达式的写法主要有下面几种

无参数,无返回值

开头的例子中就是一个无参数和无返回值的写法,在这个例子中,接口中定 义的抽象方法没有入参也没有返回值:

public class AnonymousDemo {

public static void main(String[] args) { Person person = () -> System.out.println("anonymous say hello"); person.sayHello(); }}

有参数,无返回值

先写一个这样的接口:

public interface Student {

void getAge(int age); }

在以前的代码中,需要首先写一个类去继承这个接口,或者是写一个匿名内部类, 如:

Student student=new Student() {

@Override

public void getAge(int age) { System.out.println(age); } };

使用 Lambda 就变得简单了

Student student2=(age) -> System.out.println(age);

如果只有一个参数,小括号可以不写

Student student3=age -> System.out.println(age);

如果想要在 Lambda 表达式中写一些复杂的方法,在语法上可以在大括号内部 写实现方法。

public class Main {

public static void main(String[] args) { Student student4 = age -> {

// do something

System.out.println(age); }; student4.getAge(10); } }

上面的这段代码最终会输出 10。

有参数,有返回值

首先写一个有参数,有返回值的接口方法

public interface Student {

int getAge(int age); }使用 Lambda 表达式时代码如下

public class Main {

public static void main(String[] args) { Student student = age -> {

return age * 5;};

int age = student.getAge(10); System.out.println(age); } }

在上面的代码中,由于 Student 的 getAge 方法是有返回值的,因此在写带有 大括号的实现方法时也必须带上 return。 如果实现方法用单行代码就能写完,可以将大括号和 return 进行简化,代码就 变成了下面这样:

public class Main {

public static void main(String[] args) { Student student = age -> age * 5;

int age = student.getAge(10); System.out.println(age); } }

2.4 Lambda 表达式的变量作用域

Lambda 内部如果使用了外部定义的局部变量,那这个变量是一定不能被修改的, 看下面这段代码

public class AnonymousDemo {

public static void main(String[] args) {

int num = 1; Person person = ()->{

int a = num; }; num++; } }

在 Lambda 表达式中使用了 num,又在外部对 num 进行了修改,就会出现下 面这段报错

Variable used in lambda expression should be final or effectively final

不过如果这个变量是对象变量或者静态变量,那么就没有不能修改的限制了。

public class AnonymousDemo {

static int num = 1;

public static void main(String[] args) { Person person = ()->{

int a = num; }; num++; }}

2.5 Lambda 表达式在实际中的应用

在 JDK8 中,有许多接口都可以使用 Lambda 表达式来简化代码的编写,接下 来介绍几个常见 Lambda 的使用场景。

Runnable 接口

从 Runnable 接口的源码中可以看到,这个接口是用 @FunctionalInterface 修 饰的,意味着它可以使用 Lambda 表达式。 普通的匿名类写法是这样的:

public class AnonymousDemo {

public static void main(String[] args) { Runnable runnable = new Runnable() {

@Override

public void run() { System.out.println("do something"); } }; Thread thread = new Thread(runnable); thread.start(); } }

使用 Lambda 表达式后,写法就变成了下面这样

public class AnonymousDemo {

public static void main(String[] args) { Runnable runnable = () -> System.out.println("do something"); Thread thread = new Thread(runnable); thread.start(); } }

如果再精简一点,使用一行代码就可以搞定

public class AnonymousDemo {

public static void main(String[] args) {

new Thread(() -> System.out.println("do something")).start(); } }

Comparator 接口

Comparator 是一个用来排序的接口,比如可以对 List 集合进行自定义排序:

public class AnonymousDemo {

public static void main(String[] args) { List list = Arrays.asList(3,4,2,1,6,7,9); Collections.sort(list, new Comparator() {

@Override

public int compare(Integer i1, Integer i2) {

return i1.compareTo(i2); } }); } }

而使用 Lambda 写法就会让上面的代码变得十分简单。

public class AnonymousDemo {

public static void main(String[] args) { List list = Arrays.asList(3,4,2,1,6,7,9); Collections.sort(list, (i1, i2) -> i1.compareTo(i2)); } }

2.6 实验总结

总结来讲,Lambda 表达式的使用会让代码看起来更加清晰,最好的学习方式就是实践了, 对着上面的例子多动手敲一些代码,会使你对 Lambda 的使用变得炉火纯青。

有关一文带你攻克JDK新特性的更多相关文章

  1. IDEA 2023.1 正式发布,新特性简介 - 2

     昨晚看到IDEA官推宣布IntelliJIDEA2023.1正式发布了。简单看了一下,发现这次的新版本包含了许多改进,进一步优化了用户体验,提高了便捷性。至于是否升级最新版本完全是个人意愿,如果觉得新版本没有让自己感兴趣的改进,完全就不用升级,影响不大。软件的版本迭代非常正常,正确看待即可,不持续改进就会慢慢被淘汰!根据官方介绍:IntelliJIDEA2023.1针对新的用户界面进行了大量重构,这些改进都是基于收到的宝贵反馈而实现的。官方还实施了性能增强措施,使得Maven导入更快,并且在打开项目时IDE功能更早地可用。由于后台提交检查,新版本提供了简化的提交流程。IntelliJIDEA

  2. 一文解决关于VLAN所有的疑惑 - 2

    一文解决关于VLAN所有的疑惑VLAN基本概念为什么需要VLAN?怎么在交换机上划分VLAN,VLAN的工作原理有了子网,已经隔离了广播,还需要VLAN干啥?只进行子网划分,不进行VLAN划分VLAN划分与子网划分附加VLAN信息的方法VLAN划分交换机的端口类型(Access和Trunk)一、访问链接二、汇聚链接汇聚链接VLAN间通信为什么要进行VLAN间通信?路由器实现VLAN间通信路由器和交换机的连接方式通信细节三层交换机实现VLAN间通信加速VLAN间通信三层交换机与路由器三层交换机路由器路由器和交换机配合构建LAN的实例使用VLAN设计局域网的特点VLAN增加网络的灵活性不使用VLA

  3. 一文让你彻底掌握操作符(超详细教程) - 2

    ✅作者简介:大家好,我是小杨📃个人主页:「小杨」的csdn博客🔥系列专栏:小杨带你玩转C语言【初阶】🐳希望大家多多支持🥰一起进步呀!大家好呀!我是小杨。小杨花几天的时间将C语言中的操作符这部分知识做了一个大总结,在方便自己复习的同时也能够帮助到大家。通篇字数在一万字左右,可以算作是非常详细了,一文就可以带领大家彻底掌握操作符这部分内容,文章很长建议先收藏再看,防止下次想看就找不到啦。文章目录✍1,算术操作符✍2,移位操作符    🔍2.1,左移操作符    🔍2.2,右移操作符       ✨2.2.1,算术移位       ✨2.2.2,逻辑移位✍3,位操作符    🔍3.1,按位与&   

  4. Vue3的新特性 - 2

    Vue3的新特性包括:CompositionAPI:一种新的API风格,可将有关组件功能的代码逻辑封装在单独的函数中,从而更好地管理和重用代码。Teleport:可以让组件在DOM层次结构中的任何位置渲染。Suspense:一种新的异步渲染模式,可以优化应用程序的性能。更快的渲染速度:Vue3使用了新的虚拟DOM算法,并且对渲染过程进行了优化,因此在渲染大型应用时性能更高。更小的包大小:Vue3的打包大小比Vue2更小,因为它不再需要依赖像vue-template-compiler这样的工具。其他改进:Vue3还具有其他一些改进,例如更好的TypeScript支持、更好的错误提示和更好的调试工

  5. 一文掌握软件项目成本预算、估算的方法和成本控制的秘籍 - 2

    每个企业都希望在完成项目后获得盈利,但不少企业到了年终后才发现项目做了不少,公司却并没能达到预期,甚至还出现了亏损。那么钱究竟去了哪里?很多公司都搞不清楚原因,出现糊涂账较多的状况,这将会造成严重的后果,尤其在疫情影响下,大环境很恶劣,如果是大公司的事业部门出现亏损,就可能会导致事业部门解散;如果是小公司出现亏损,就很容易导致公司倒闭;怎样做才能确保我们所完成的项目都能获利?从财务角度看,要确保盈利必须做到合理估算成本,只有这样才能在对外签订合约时做出合理报价,在对内在开始项目前做出充分评估投入代价,同时在实施过程中还要控制成本得当,最后项目结束时才会有可能获得盈利。那么我们怎样才能准确的判断

  6. 一文详解COINDAO是什么? - 2

    COINDAO旨在重建社区信任和安全。基于皖北基因的强烈共识,COINDAO自发产生了一个共创、共建、共治、共享的协作组织。它专注于DAO投资管理协议,为新的优质项目创造增长技术和资金。COINDAO的使命就是为真正的优质项目打造一个去中心化、公开透明的平台,让各个赛道上的优质项目能够以更低的成本快速募集资金并向公众开放.打破头部垄断。让真正的爱好者直接获得早期参与优质项目的资格,不再遥不可及,构建生态应用的可信体系,打造人人参与共建、人人共享的去中心化DAO好处。生态系统,我们称之为COINDAO生态系统。COINDAO国内各大财经网站宣发如下:COINDAO国外各大财经网站宣发: COIN

  7. 一文吃透前端低代码的 “神仙生活” - 2

    今天来说说前端低代码有多幸福?低代码是啥?顾名思义少写代码……这种情况下带来的幸福有:代码写得少,bug也就越少(所谓“少做少错”),因此开发环节的两大支柱性工作“赶需求”和“修bug”就都少了;要测的代码少了,那么测试用例也可以少写了。所以,总结低代码带来的幸福感有这三大点:开发效率提高开发成本减少维护性更高针对上述三点,我们展开说说。01、开发效率提高对于低代码的理解,个人认为可以通过配置化的低成本交互方式(主流是拖拽)加上少量的胶水代码,去满足一类应用的需求。这就说明,基于低代码,开发人员无需代码或说只需少量代码就可以开发出各类应用管理系统,如:OA协同办公、KM知识管理、CRM客户关系

  8. javascript - 如何在 Netbeans 的 Nashorn (JDK 8+) 中调试 JavaScript? - 2

    我看到了最新的Netbeans8.0beta提供了一种调试“在Nashorn中执行的JavaScript代码”的方法(参见附件)查看此操作的步骤是什么?[更新]我只能通过打开文件(在任何项目之外)运行一个简单的JavaScript文件,在文件内部单击,右键单击,然后选择“运行文件”选项。我需要指出快捷方式“Shift-F6”对我不起作用。因此,如果其他人知道运行它的不同方式,请发布您的解决方案。[更新2]关于JavaScriptNashorn的新文章提及Netbeans中的JavaScript调试[更新3]似乎Nashhorn将在JDK内部被弃用。参见JEP335:Deprecatet

  9. 【C语言进阶】还说不会?一文带你全面掌握计算机预处理操作 - 2

    目录🍊前言🍊:🍈一、宏与函数🍈:        1.宏与函数对比:    2.宏与函数的命名约定:🍓二、预处理操作符🍓:    1.预处理操作符"#":    2.预处理操作符"##":🥝三、条件编译🥝:    1.简述条件编译指令:    2.常见条件编译指令:🍒总结🍒:🛰️博客主页:✈️銮同学的干货分享基地🛰️欢迎关注:👍点赞🙌收藏✍️留言🛰️系列专栏:💐【进阶】C语言学习            🧧  C语言学习🛰️代码仓库:🎉VS2022_C语言仓库    家人们更新不易,你们的👍点赞👍和⭐关注⭐真的对我真重要,各位路过的友友麻烦多多点赞关注,欢迎你们的私信提问,感谢你们的转发!    

  10. 一文带你通俗理解23种软件设计模式(推荐收藏,适合小白学习,附带C++例程完整源码) - 2

    作者:翟天保Steven版权声明:著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处一、设计模式是什么?    设计模式是为了解决在软件开发过程中遇到的某些问题而形成的思想。同一场景有多种设计模式可以应用,不同的模式有各自的优缺点,开发者可以基于自身需求选择合适的设计模式,去解决相应的工程难题。    良好的软件设计和架构,可以让代码具备良好的可读性、可维护性、可扩展性、可复用性,让整个系统具备较强的鲁棒性和性能,减少屎山代码出现的概率。    想要熟练运用设计模式,提高自己的编程能力和架构能力,只有在自己工作中,结合自身工作内容,多思考多实践。本文只能通过举一些通俗的例子,来

随机推荐