摘要:欧阳思海继承接口后,又加了新的抽象方法,这个接口就不再是函数式接口默认方法在接口中添加了一个默认方法。总结在这篇文章中,我们讲了表达式方法引用函数式接口接口中的静态方法接口中的默认方法的使用。
今天我来聊聊 Java8 的一些新的特性,确实 Java8 的新特性的出现,给开发者带来了非常大的便利,可能刚刚开始的时候会有点不习惯的这种写法,但是,当你真正的熟悉了之后,你一定会爱上这些新的特性的,这篇文章就来聊聊这些新特性。
lambda 表达式lambda 表达式在项目中也是用到了,这种新的语法的加入,对于使用 Java 多年的我,我觉得是如虎添翼的感觉哈,这种新的语法,大大的改善了以前的 Java 的代码,变得更加的简洁,我觉得这也是为什么 Java8 能够很快的流行起来的原因吧。
这里我们用几个以前的经典的 Java 的写法和用 lambda 表达式的方式进行对比。
线程的用法
原始的线程用法
//使用匿名内部类的方式启动多线程 new Thread(new Runnable() { @Override public void run() { System.out.println("这是使用匿名内部类的方式。。。"); } }).start();
lambda 表达式
//使用lambda表达式方式 new Thread(() -> { System.out.println("这是使用lambda表达式的方式。。。"); }).start();
你会发现,用 lambda 表达式的方式能够写更少的代码,看起来也会更加的舒服和简洁。
这里没有使用参数,只是一个简单的例子。
我们再看一个例子。
遍历方式
原始方式
//原始方式 Listlist = Arrays.asList(1, 2, 3, 4, 5); for (int i : list) { System.out.println(i); }
lambda 表达式方式
//使用lambda表达式代替foreach循环 Stream.of(1, 2, 3, 4, 5).forEach((x) -> { System.out.println(x); });
在原始的方式中,我们一般使用 foreach 的方式进行遍历,有了 Java8 的方式之后,我们可以用 forEach 方法,然后,再用 lambda 表达式的方式进行遍历,也让原来的方式变得更加的简洁。
在这个例子中,我们加了一个参数,在()中间我们加了一个 x ,代表的意思其实是:通过 forEach 方法,我们把一个元素已经赋值到 x 中了,拿到这个 x ,我们就可以输出结果。
总结
lambda 的使用方式其实很简单,可以总结为下面的方法。
([参数可选,...]) -> { }方法引用
方法引用其实是 lambda 表达式的部分的简化,也就是为了简化 lambda 表达式而存在的感觉,下面我们还讲讲怎么使用方法引用。
/** * @return void * @Author ouyangsihai * @Description 方法引用测试 * @Date 10:23 2019/5/14 * @Param [] **/ @Test public void test_method_reference() { //使用lambda表达式 Stream.of("A", "BB", "CCC", "DDDD", "FFFFF") .map(s -> s.length()) //lambda .forEach((x) -> { System.out.println(x); }); //使用静态方法引用 Stream.of("A", "BB", "CCC", "DDDD", "FFFFF") .map(String::length) //静态方法引用 .forEach((x) -> { System.out.println(x); }); //使用实例方法引用 Stream.of( new ClassMate("1", "欧阳思海"), new ClassMate("2", "sihai") ).map(ClassMate::getName)//实例方法引用 .forEach(x -> { System.out.println(x); }); }
在第一个测试中,我们用的是 lambda 表达式来获取每个字符串的长度。
s -> s.length()
在第二个测试中,我们使用的是静态方法引用来获取每个字符串的长度。
String::length
在第三个测试中,我们使用的是实例方法引用。
ClassMate::getName
解释
① map 方法是映射的意思。
② forEach 方式是遍历每一个元素。
③ ClassMate 是一个包含 id 和 name 的简单 po 类。
通过上面这个例子,基本上我们就知道怎么使用方法引用了。下面我们进行一个小的总结。
总结
① 使用方法
类名::方法名
② 方法可以是:静态方法,实例方法
构造函数引用在上面我们讲了方法引用的基本使用方法,其实除了方法引用以外,还有构造函数引用,回想一下,以前我们创建对象是怎么做?是不是需要 new 一个对象呢,那么现在用构造函数引用又是怎么做的呢?
下面我们用一个例子讲解一下,在这个例子中,对象还是使用上面的 ClassMate。
/** * @return void * @Author ouyangsihai * @Description 构造函数引用测试 * @Date 10:23 2019/5/14 * @Param [] **/ @Test public void test_method_reference2() { //使用lambda表达式 Stream.of("A", "BB", "CCC", "DDDD", "FFFFF") .map(s -> new ClassMate(s)) //lambda .collect(Collectors.toList()); //使用构造函数引用 Stream.of("A", "BB", "CCC", "DDDD", "FFFFF") .map(ClassMate::new) //构造函数引用,由上下文决定用哪一个构造函数 .collect(Collectors.toList()); }
① 第一个我们使用的是 lambda 表达式进行创建对象的 s -> new ClassMate(s)。
② 第二个我们使用的是构造函数引用创建对象的 ClassMate::new 。
③ 我们发现构造函数引用:类名::new ,然后对于使用哪一个构造函数是由上下文决定的,比如有一个参数和两个参数和无参数的构造函数,会自动确定用哪一个。
在 Java 8 之前的接口是不能有实现的,只能定义抽象方法,然而,在 Java 8 以后,增加了一个新的功能,可以添加实现,可以定义默认方法,可以定义静态方法。
函数式接口什么是函数式接口呢?
这个名词在 Java 中以前是很少听到的,但是正是有了 Java 8 的横空出世,函数式编程也变得熟悉了。
在一个接口中我们以 @FunctionalInterface 注解声明一个接口,并且接口中只有一个抽象方法,那么我们就叫做这是一个函数式接口。
/** * @ClassName FunctionalInterfaceTest * @Description * @Author 欧阳思海 * @Date 2019/5/14 10:39 * @Version 1.0 **/ @FunctionalInterface public interface FunctionalInterfaceTest { //继承接口后,又加了新的抽象方法,这个接口就不再是函数式接口 void test(String s); }
① 上面的接口中只有一个抽象方法,所以这是一个函数式接口。
② 如果上面接口中再加一个抽象方法,那么就不是函数式接口了。
下面,我们再通过继承来继承这个接口。
/** * @ClassName FunctionalTest * @Description * @Author 欧阳思海 * @Date 2019/5/17 17:26 * @Version 1.0 **/ public interface FunctionalTest extends FunctionalInterfaceTest{ int test2(); }
① 我们继承了上面的接口,并且加了一个 test2 方法。
② 这里注意,如果一个接口集成现有的函数式接口后,又加了其他的抽象方法,这个接口就不是函数式接口了。
默认方法很简单,用 default 声明即可。
/** * @ClassName FunctionalInterfaceTest * @Description * @Author 欧阳思海 * @Date 2019/5/14 10:39 * @Version 1.0 **/ @FunctionalInterface public interface FunctionalInterfaceTest { //继承接口后,又加了新的抽象方法,这个接口就不再是函数式接口 void test(String s); //默认方法 default String getStr(){ return null; } }
① 在接口中添加了一个默认方法。并且实现了方法。
静态方法默认方法很简单,用 static 声明即可。
/** * @ClassName FunctionalInterfaceTest * @Description * @Author 欧阳思海 * @Date 2019/5/14 10:39 * @Version 1.0 **/ @FunctionalInterface public interface FunctionalInterfaceTest { //继承接口后,又加了新的抽象方法,这个接口就不再是函数式接口 void test(String s); //静态方法 static String getStr2(){ return null; } //错误用法 default static String getStr3(){ return null; } }
① 实现的静态方法,用 static 声明。
② 注意不能同时使用 default 和 static 声明。
在这篇文章中,我们讲了 lambda 表达式、方法引用、函数式接口、接口中的静态方法、接口中的默认方法的使用。
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/74858.html
摘要:上一篇小乐介绍了新特性函数式接口,大家可以点击回顾。中引入方法引用新特性用于简化应用对象方法的调用,方法引用是用来直接访问类或者实例的已经存在的方法或者构造方法。方法引用是一种更简洁易懂的表达式。 上一篇小乐介绍了《Java8新特性-函数式接口》,大家可以点击回顾。这篇文章将接着介绍Java8新特性之方法引用。 Java8 中引入方法引用新特性,用于简化应用对象方法的调用, 方法引用是...
摘要:大家好,我是乐字节的小乐,上一次我们说到了核心特性之函数式接口,接下来我们继续了解又一核心特性方法引用。方法引用是一种更简洁易懂的表达式。感谢光临阅读小乐的,敬请关注乐字节后续将继续讲述等前沿知识技术。 大家好,我是乐字节的小乐,上一次我们说到了Java8核心特性之函数式接口,接下来我们继续了解Java8又一核心特性——方法引用。 showImg(https://segmentfaul...
摘要:实际上方法引用是表达式的一种语法糖。小结本篇全面介绍了方法引用的四种使用方式,且每种方式都有对应一个示例来帮助大家理解。 上一篇我们详细介绍了Optional类用来避免空指针问题,本篇我们全面了解一下Java8中的方法引用特性。方法引用是lambda表达式的一种特殊形式,如果正好有某个方法满足一个lambda表达式的形式,那就可以将这个lambda表达式用方法引用的方式表示,但是如果这...
摘要:一表达式匿名内部类最大的问题在于其冗余的语法,比如前面的中五行代码仅有一行是在执行任务。总结基于词法作用域的理念,表达式不可以掩盖任何其所在上下文的局部变量。 转载请注明出处:https://zhuanlan.zhihu.com/p/20540175 在介绍Lambda表达式之前,我们先来看只有单个方法的Interface(通常我们称之为回调接口): public interface...
摘要:表达式体现了函数式编程的思想,即一个函数亦可以作为另一个函数参数和返回值,使用了函数作参数返回值的函数被称为高阶函数。对流对象进行及早求值,返回值不在是一个对象。 Java8主要的改变是为集合框架增加了流的概念,提高了集合的抽象层次。相比于旧有框架直接操作数据的内部处理方式,流+高阶函数的外部处理方式对数据封装更好。同时流的概念使得对并发编程支持更强。 在语法上Java8提供了Lamb...
阅读 3384·2021-11-25 09:43
阅读 3447·2021-11-19 09:40
阅读 2384·2021-10-14 09:48
阅读 1260·2021-09-09 11:39
阅读 1903·2019-08-30 15:54
阅读 2802·2019-08-30 15:44
阅读 1977·2019-08-29 13:12
阅读 1515·2019-08-29 12:59