java中lambda表达式有专业的文档吗?

2020-07-17 11:54发布

1条回答
向笑 - 活到老,学到老!
2楼 · 2020-07-23 09:07

为什么要引入Lambda表达式

简单的来说,引入Lambda就是为了简化代码,允许把函数作为一个方法的参数传递进方法中。如果有JavaScript的编程经验,马上会想到这不就是闭包吗。是的,Lambda表达式也可以称作Java中的闭包。

先回顾一下Java8以前,如果想把某个接口的实现类作为参数传递给一个方法会怎么做?要么创建一个类实现该接口,然后new出一个对象,在调用方法时传递进去,要么使用匿名类,可以精简一些代码。以创建一个线程并打印一行日志为例,使用匿名函数写法如下:

  1. new Thread(new Runnable() {

  2.    @Override

  3.    public void run() {

  4.        System.out.println("欢迎关注公众号:程序新视界");

  5.    }

  6. }).start();

在java8以前,使用匿名函数已经算是很简洁的写法了,再来看看使用Lambda表达式,上面的代码会变成什么样子。

  1. new Thread(() -> System.out.println("欢迎关注公众号:程序新视界")).start();

是不是简洁到爆!

我们都知道java是面向对象的编程语言,除了部分简单数据类型,万物皆对象。因此,在Java中定义函数或方法都离不开对象,也就意味着很难直接将方法或函数像参数一样传递,而Java8中的Lambda表达式的出现解决了这个问题。

Lambda表达式使得Java拥有了函数式编程的能力,但在Java中Lambda表达式是对象,它必须依附于一类特别的对象类型——函数式接口(functional interface),后面详细讲解。

Lambda表达式简介

Lambda表达式是一种匿名函数(对Java而言这并不完全准确),通俗的说,它是没有声明的方法,即没有访问修饰符、返回值声明和名字的方法。使用Lambda表达式的好处很明显就是可以使代码变的更加简洁紧凑。

Lambda表达式的使用场景与匿名类的使用场景几乎一致,都是在某个功能(方法)只使用一次的时候。

Lambda表达式语法结构

Lambda表达式通常使用(param)->(body)语法书写,基本格式如下:

  1. //没有参数

  2. () -> body

  3. // 1个参数

  4. (param) -> body

  5. // 或

  6. (param) ->{ body; }

  7. // 多个参数

  8. (param1, param2...) -> { body }

  9. // 或

  10. (type1 param1, type2 param2...) -> { body }

常见的Lambda表达式如下:

  1. // 无参数,返回值为字符串“公众号:程序新视界”

  2. () -> "公众号:程序新视界";

  3. // 1个String参数,直接打印结果

  4. (System.out::println);

  5. // 或

  6. (String s) -> System.out.print(s)

  7. // 1个参数(数字),返回2倍值  

  8. x -> 2 * x;

  9. // 2个参数(数字),返回差值

  10. (x, y) -> x – y

  11. // 2个int型整数,返回和值  

  12. (int x, int y) -> x + y

对照上面的示例,我们再总结一下Lambda表达式的结构:

  • Lambda表达式可以有0~n个参数。

  • 参数类型可以显式声明,也可以让编译器从上下文自动推断类型。如(int x)和(x)是等价的。

  • 多个参数用小括号括起来,逗号分隔。一个参数可以不用括号。

  • 没有参数用空括号表示。

  • Lambda表达式的正文可以包含零条,一条或多条语句,如果有返回值则必须包含返回值语句。如果只有一条可省略大括号。如果有一条以上则必须包含在大括号(代码块)中。

函数式接口

函数式接口(Functional Interface)是Java8对一类特殊类型的接口的称呼。这类接口只定义了唯一的抽象方法的接口(除了隐含的Object对象的公共方法),因此最开始也就做SAM类型的接口(Single Abstract Method)。

比如上面示例中的java.lang.Runnable就是一种函数式接口,在其内部只定义了一个void run()的抽象方法,同时在该接口上注解了@FunctionalInterface。

  1. @FunctionalInterface

  2. public interface Runnable {

  3.    public abstract void run();

  4. }

@FunctionalInterface注解是用来表示该接口要符合函数式接口的规范,除了隐含的Object对象的公共方法以外只可有一个抽象方法。当然,如果某个接口只定义一个抽象方法,不使用该注解也是可以使用Lambda表达式的,但是没有该注解的约束,后期可能会新增其他的抽象方法,导致已经使用Lambda表达式的地方出错。使用@FunctionalInterface从编译层面解决了可能的错误。

比如当注解@FunctionalInterface之后,写两个抽象方法在接口内,会出现以下提示:

  1. Multiple non-overriding abstract methods found in interface com.secbro2.lambda.NoParamInterface

通过函数式接口我们也可以得出一个简单的结论:可使用Lambda表达式的接口,只能有一个抽象方法(除了隐含的Object对象的公共方法)。

注意此处的方法限制为抽象方法,如果接口内有其他静态方法则不会受限制。

方法引用,双冒号操作

[方法引用]的格式是,类名::方法名。

像如ClassName::methodName或者objectName::methodName的表达式,我们把它叫做方法引用(Method Reference),通常用在Lambda表达中。

看一下示例:

  1. // 无参数情况

  2. NoParamInterface paramInterface2 = ()-> new HashMap<>();

  3. // 可替换为

  4. NoParamInterface paramInterface1 = HashMap::new;

  5. // 一个参数情况

  6. OneParamInterface oneParamInterface1 = (String string) -> System.out.print(string);

  7. // 可替换为

  8. OneParamInterface oneParamInterface2 = (System.out::println);

  9. // 两个参数情况

  10. Comparator c = (Computer c1, Computer c2) -> c1.getAge().compareTo(c2.getAge());

  11. // 可替换为

  12. Comparator c = (c1, c2) -> c1.getAge().compareTo(c2.getAge());

  13. // 进一步可替换为

  14. Comparator c = Comparator.comparing(Computer::getAge);

再比如我们用函数式接口java.util.function.Function来实现一个String转Integer的功能,可以如下写法:

  1. Function function = Integer::parseInt;

  2. Integer num = function.apply("1");

根据Function接口的定义Function,其中T表示传入类型,R表示返回类型。具体就是实现了Function的apply方法,在其方法内调用了Integer.parseInt方法。

通过上面的讲解,基本的语法已经完成,以下内容通过实例来逐一演示在不同的场景下如何使用。

Runnable线程初始化示例

Runnable线程初始化是比较典型的应用场景。

  1. // 匿名函类写法

  2. new Thread(new Runnable() {

  3.    @Override

  4.    public void run() {

  5.        System.out.println("欢迎关注公众号:程序新视界");

  6.    }

  7. }).start();

  8. // lambda表达式写法

  9. new Thread(() -> System.out.println("欢迎关注公众号:程序新视界")).start();

  10. // lambda表达式 如果方法体内有多行代码需要带大括号

  11. new Thread(() -> {

  12.    System.out.println("欢迎关注公众号");

  13.    System.out.println("程序新视界");

  14. }).start();

通常都会把lambda表达式内部变量的名字起得短一些,这样能使代码更简短。

事件处理示例

Swing API编程中经常会用到的事件监听。

  1. // 匿名函类写法

  2. JButton follow =  new JButton("关注");

  3. follow.addActionListener(new ActionListener() {

  4.    @Override

  5.    public void actionPerformed(ActionEvent e) {

  6.        System.out.println("已关注公众号:程序新视界");

  7.    }

  8. });

  9. // lambda表达式写法

  10. follow.addActionListener((e) -> System.out.println("已关注公众号:程序新视界"));

  11. // lambda表达式写法

  12. follow.addActionListener((e) -> {

  13.    System.out.println("已关注公众号");

  14.    System.out.println("程序新视界");

  15. });

列表遍历输出示例

传统遍历一个List,基本上都使用for循环来遍历,Java8之后List拥有了forEach方法,可配合lambda表达式写出更加简洁的方法。

  1. List list = Arrays.asList("欢迎","关注","程序新视界");

  2. // 传统遍历

  3. for(String str : list){

  4.    System.out.println(str);

  5. }

  6. // lambda表达式写法

  7. list.forEach(str -> System.out.println(str));

  8. // lambda表达式写法

  9. list.forEach(System.out::println);

函数式接口示例

在上面的例子中已经看到函数式接口java.util.function.Function的使用,在java.util.function包下中还有其他的类,用来支持Java的函数式编程。比如通过Predicate函数式接口以及lambda表达式,可以向API方法添加逻辑,用更少的代码支持更多的动态行为。

  1. @Test

  2. public void testPredicate() {

  3.    List list = Arrays.asList("欢迎", "关注", "程序新视界");

  4.    filter(list, (str) -> ("程序新视界".equals(str)));

  5.    filter(list, (str) -> (((String) str).length() == 5));

  6. }

  7. public static void filter(List list, Predicate condition) {

  8.    for (String content : list) {

  9.        if (condition.test(content)) {

  10.            System.out.println("符合条件的内容:" + content);

  11.        }

  12.    }

  13. }

其中filter方法中的写法还可以进一步简化:

  1. list.stream().filter((content) -> condition.test(content)).forEach((content) ->System.out.println("符合条件的内容:" + content));

  2. list.stream().filter(condition::test).forEach((content) ->System.out.println("符合条件的内容:" + content));

  3. list.stream().filter(condition).forEach((content) ->System.out.println("符合条件的内容:" + content));

如果不需要“符合条件的内容:”字符串的拼接,还能够进一步简化:

  1. list.stream().filter(condition).forEach(System.out::println);

如果将调用filter方法的判断条件也写在一起,test方法中的内容可以通过一行代码来实现:

  1. list.stream().filter((str) -> ("程序新视界".equals(str))).forEach(System.out::println);

如果需要同时满足两个条件或满足其中一个即可,Predicate可以将这样的多个条件合并成一个。

  1. Predicate start = (str) -> (((String) str).startsWith("程序"));

  2. Predicate len = (str) -> (((String) str).length() == 5);

  3. list.stream().filter(start.and(len)).forEach(System.out::println);

Stream相关示例

在《JAVA8 STREAM新特性详解及实战》一文中已经讲解了Stream的使用。你是否发现Stream的使用都离不开Lambda表达式。是的,所有Stream的操作必须以Lambda表达式为参数。

以Stream的map方法为例:

  1. Stream.of("a","b","c").map(item -> item.toUpperCase()).forEach(System.out::println);

  2. Stream.of("a","b","c").map(String::toUpperCase).forEach(System.out::println);

更多的使用实例可参看Stream的《JAVA8 STREAM新特性详解及实战》一文。

Lambda表达式与匿名类的区别

  • 关键词的区别:对于匿名类,关键词this指向匿名类,而对于Lambda表达式,关键词this指向包围Lambda表达式的类的外部类,也就是说跟表达式外面使用this表达的意思是一样。

  • 编译方式:Java编译器编译Lambda表达式时,会将其转换为类的私有方法,再进行动态绑定,通过invokedynamic指令进行调用。而匿名内部类仍然是一个类,编译时编译器会自动为该类取名并生成class文件。

其中第一条,以Spring Boot中ServletWebServerApplicationContext类的一段源码作为示例:

  1. private org.springframework.boot.web.servlet.ServletContextInitializer getSelfInitializer() {

  2.   return this::selfInitialize;

  3. }

  4. private void selfInitialize(ServletContext servletContext) throws ServletException {

  5.   prepareWebApplicationContext(servletContext);

  6.   registerApplicationScope(servletContext);

  7.   WebApplicationContextUtils.registerEnvironmentBeans(getBeanFactory(),servletContext);

  8.   for (ServletContextInitializer beans : getServletContextInitializerBeans()) {

  9.      beans.onStartup(servletContext);

  10.   }

  11. }

其中,这里的this指向的就是getSelfInitializer方法所在的类。


相关问题推荐

  • 回答 2

    Statement的execute(String query)方法用来执行任意的SQL查询,如果查询的结果是一个ResultSet,这个方法就返回true。如果结果不是ResultSet,比如insert或者update查询,它就会返回false。我们可以通过它的getResultSet方法来获取ResultSet,或者通过getUpda...

  • 回答 22

    忙的时候项目期肯定要加班 但是每天加班应该还不至于

  • 回答 108
    已采纳

    虽然Java人才越来越多,但是人才缺口也是很大的,我国对JAVA工程师的需求是所有软件工程师当中需求大的,达到全部需求量的60%-70%,所以Java市场在短时间内不可能饱和。其次,Java市场不断变化,人才需求也会不断增加。马云说过,未来的制造业要的不是石油,...

  • 回答 5
    已采纳

    工信部证书含金量较高。工信部是国务院的下属结构,具有发放资质、证书的资格。其所发放的证书具有较强的权威性,在全国范围内收到认可,含金量通常都比较高。 工信部证书,其含义也就是工信部颁发并承认的某项技能证书,是具有法律效力的,并且是国家认可的...

  • 回答 70
    已采纳

    学Java好不好找工作?看学完Java后能做些什么吧。一、大数据技术Hadoop以及其他大数据处理技术都是用Java或者其他,例如Apache的基于Java 的 HBase和Accumulo以及ElasticSearchas。但是Java在此领域并未占太大空间,但只要Hadoop和ElasticSearchas能够成长壮...

  • 回答 16
    已采纳

    就是java的基础知识啊,比如Java 集合框架;Java 多线程;线程的五种状态;Java 虚拟机;MySQL (InnoDB);Spring 相关;计算机网络;MQ 消息队列诸如此类

  • 回答 12

    #{}和${}这两个语法是为了动态传递参数而存在的,是Mybatis实现动态SQL的基础,总体上他们的作用是一致的(为了动态传参),但是在编译过程、是否自动加单引号、安全性、使用场景等方面有很多不同,下面详细比较两者间的区别:1.#{} 是 占位符 :动态解析 ...

  • 回答 62

    没问题的,专科学历也能学习Java开发的,主要看自己感不感兴趣,只要认真学,市面上的培训机构不少都是零基础课程,能跟得上,或是自己先找些资料学习一下。

  • 回答 4

    1、反射对单例模式的破坏采用反射的方式另辟蹊径实例了该类,导致程序中会存在不止一个实例。解决方案其思想就是采用一个全局变量,来标记是否已经实例化过了,如果已经实例化过了,第 二次实例化的时候,抛出异常2、clone()对单例模式的破坏当需要实现单例的...

  • 回答 5

     优点: 一、实例控制  单例模式会阻止其他对象实例化其自己的单例对象的副本,从而确保所有对象都访问唯一实例。 二、灵活性  因为类控制了实例化过程,所以类可以灵活更改实例化过程。 缺点: 一、开销  虽然数量很少,但如果每次对象请求引用时都要...

  • 回答 4

    这个主要是看你数组的长度是多少, 比如之前写过的一个程序有个数组存的是各个客户端的ip地址:string clientIp[4]={XXX, xxx, xxx, xxx};这个时候如果想把hash值对应到上面四个地址的话,就应该对4取余,这个时候p就应该为4...

  • 回答 6

     哈希表的大小 · 关键字的分布情况 · 记录的查找频率 1.直接寻址法:取关键字或关键字的某个线性函数值为散列地址。即H(key)=key或H(key) = a·key + b,其中a和b为常数(这种散列函数叫做自身函数)。...

  • 回答 6

    哈希表的大小取决于一组质数,原因是在hash函数中,你要用这些质数来做模运算(%)。而分析发现,如果不是用质数来做模运算的话,很多生活中的数据分布,会集中在某些点上。所以这里最后采用了质数做模的除数。 因为用质数做了模的除数,自然存储空间的大小也用质数了...

  • 回答 2

    是啊,哈希函数的设计至关重要,好的哈希函数会尽可能地保证计算简单和散列地址分布均匀,但是,我们需要清楚的是,数组是一块连续的固定长度的内存空间

  • 回答 3

     解码查表优化算法,seo优化

  • 回答 5

    1.对对象元素中的关键字(对象中的特有数据),进行哈希算法的运算,并得出一个具体的算法值,这个值 称为哈希值。2.哈希值就是这个元素的位置。3.如果哈希值出现冲突,再次判断这个关键字对应的对象是否相同。如果对象相同,就不存储,因为元素重复。如果对象不同,就...

没有解决我的问题,去提问