Lambda


一、概述

 

1、什么是Lambda表达式

  1. //1.不需要参数,返回值为5
  2. () -> 5
  3.  
  4. //2.接收一个参数(数字类型),返回其2倍的值
  5. x -> 2 * x
  6.  
  7. //3.接收2个参数(数字),并返回他们的差值
  8. x,y)-> x-y
  9.  
  10. //4.接收2个int类型整数,返回他们的和
  11. int x, int y -> x+y
  12.  
  13. //5.接受一个String对象,并在控制台打印,不返回任何值(看起来像是返回void)
  14. (String s) ->System.out.print(s)

  

3.什么是函数式接口

  在对上面进行举例说明之前,必须先来理解一下函数式接口,因为Lambda是建立在函数式接口的基础上的

  记住!

  (1)只包含一个抽象方法的接口,称为函数式接口

  1. public class Lambda01 {
  2. public static void main(String[] args) {
  3. new Thread(new Runnable() {
  4. @Override
  5. public void run() {
  6. System.out.println("线程1开启"+":"+Thread.currentThread().getName());
  7. }
  8. }).start();
  9.  
  10. new Thread(() -> System.out.println("线程2开启"+":"+Thread.currentThread().getName())).start();
  11. Runnable runnable1= new Runnable(){
  12. @Override
  13. public void run() {
  14. System.out.println("线程3开启"+":"+Thread.currentThread().getName());
  15. }
  16. };
  17. Runnable runnable2=()-> System.out.println("线程4开启"+":"+Thread.currentThread().getName());
  18. runnable1.run();
  19. runnable2.run();
  20.  
  21. }
  22. }
  23. /*输出结果
  24. * Hello world !
  25. * Hello world !
  26. * Hello world !
  27. * Hello world !
  28. */

通过上面案例可以看出:通过Lambda表达式看去舒服清爽多了,2而通过匿名内部类代码总是不够整洁。

  1. public class TestArray {
  2.  
  3. public static void main(String[] args) {
  4. String[] players = {"zhansgan", "lisi", "wangwu", "zhaoliu", "wangmazi"};
  5.  
  6. // 1.1 使用匿名内部类根据 surname 排序 players
  7. Arrays.sort(players, new Comparator<String>() {
  8. @Override
  9. public int compare(String s1, String s2) {
  10. return (s1.compareTo(s2));
  11. }
  12. });
  13.  
  14. // 1.2 使用 lambda 排序,根据 surname
  15. Arrays.sort(players, (String s1, String s2) -> s1.compareTo(s2));
  16.  
  17. //================================================================================================
  18.  
  19. // 2.1 使用匿名内部类根据 name lenght 排序 players
  20. Arrays.sort(players, new Comparator<String>() {
  21. @Override
  22. public int compare(String s1, String s2) {
  23. return (s1.length() - s2.length());
  24. }
  25. });
  26.  
  27. // 2.2使用Lambda,根据name length
  28. Arrays.sort(players, (String s1, String s2) -> (s1.length() - s2.length()));
  29.  
  30. //==================================================================================================
  31.  
  32. // 3.1 使用匿名内部类排序 players, 根据最后一个字母
  33. Arrays.sort(players, new Comparator<String>() {
  34. @Override
  35. public int compare(String s1, String s2) {
  36. return (s1.charAt(s1.length() - 1) - s2.charAt(s2.length() - 1));
  37. }
  38. });
  39.  
  40. // 3.2 使用Lambda,根据最后一个字母
  41. Arrays.sort(players, (String s1, String s2) -> (s1.charAt(s1.length() - 1) - s2.charAt(s2.length() - 1)));
  42. }
  43. }

通过上面例子我们再来思考为什么Lambda表达式需要函数式接口?其实很简单目的就是为来保证唯一。

你的Runnable接口只要一个抽象方法,那么我用() -> System.out.println("Hello world !"),就只能代表run方法,如果你下面还有一个抽象方法,那我使用Lambda表达式,

那鬼才知道要调用哪个抽象方法呢。

二、方法引用

  首先注意:方法引用,不是方法调用!方法引用,不是方法调用!方法引用,不是方法调用!

  函数式接口的实例可以通过lambda表达式、方法引用、构造方法引用来创建,方法引用是lambda表达式的语法糖,任何用方法引用的地方都可由lambda表达式来替换,

  但是并不是所有的lambda表达式都可以用方法引用来替换

  举例

  这就是一个打印集合所有元素的例子,value ->System.out.println(value)是一个Consumer函数式接口,这个函数式接口可以通过方法引用来替换

  1. public class TestArray {
  2.  
  3. public static void main(String[] args) {
  4. List<String> list = Arrays.asList("xuxiaoxiao", "xudada", "xuzhongzhong");
  5. list.forEach(value -> System.out.println(value));
  6. }
  7. /* 输出:
  8. * xuxiaoxiao
  9. * xudada
  10. * xuzhongzhong
  11. */
  12. }

2、分类

类别 使用形式
静态方法引用 类名 :: 静态方法名
实例方法引用 对象名(引用名) :: 实例方法名
类方法引用 类名 :: 实例方法名
构造方法引用 类名 :: new

(1)静态方法引用

  1. public class Apple {
  2.  
  3. private String name;
  4. private String color;
  5. private double weight;
  6.  
  7. public Apple(String name, String color, double weight) {
  8. this.name = name;
  9. this.color = color;
  10. this.weight = weight;
  11. }
  12.  
  13. public static int compareByWeight(Apple a1, Apple a2) {
  14. double diff = a1.getWeight() - a2.getWeight();
  15. return new Double(diff).intValue();
  16. }
  17.  
  18. //还有getter setter toString
  19. }

有一个苹果的List,现在需要根据苹果的重量进行排序。List 的 sort 函数接收一个 Comparator 类型的参数,Comparator 是一个函数式接口,接收两个参数,

返回一个int值。Apple的静态方法compareByWeight正好符合Comparator函数式接口,所以可以使用:

Apple::compareByWeight 静态方法引用来替代lambda表达式

  1. public class LambdaTest {
  2.  
  3. public static void main(String[] args) {
  4.  
  5. Apple apple1 = new Apple("红富士", "Red", 280);
  6. Apple apple2 = new Apple("冯心", "Yello", 470);
  7. Apple apple3 = new Apple("大牛", "Red", 320);
  8. Apple apple4 = new Apple("小小", "Green", 300);
  9.  
  10. List<Apple> appleList = Arrays.asList(apple1, apple2, apple3, apple4);
  11.  
  12. //lambda 表达式形式
  13. //appleList.sort((Apple a1, Apple a2) -> {
  14. // return new Double(a1.getWeight() - a2.getWeight()).intValue();
  15. //});
  16.  
  17. //静态方法引用形式(可以看出引用方法比上面的更加简单
  18. appleList.sort(Apple::compareByWeight);
  19.  
  20. appleList.forEach(apple -> System.out.println(apple));
  21.  
  22. }
  23. }
  24. 输出:
  25. Apple{category='红富士', color='Red', weight=280.0}
  26. Apple{category='小小', color='Green', weight=300.0}
  27. Apple{category='大牛', color='Red', weight=320.0}
  28. Apple{category='冯心', color='Yello', weight=470.0}

注意:Apple.compareByWeight是方法的调用,而Apple::compareByWeight方法引用,这两者完全不是一回事。

(2)实例方法引用

这个compareByWeight是一个实例方法

  1. public class AppleComparator {
  2.  
  3. public int compareByWeight(Apple a1, Apple a2) {
  4. double diff = a1.getWeight() - a2.getWeight();
  5. return new Double(diff).intValue();
  6. }
  7. }

下面的例子通过实例对象的方法引用 comparator::compareByWeight 来代替lambda表达式

  1. public class LambdaTest {
  2.  
  3. public static void main(String[] args) {
  4.  
  5. Apple apple1 = new Apple("红富士", "Red", 280);
  6. Apple apple2 = new Apple("冯心", "Yello", 470);
  7. Apple apple3 = new Apple("哈哈", "Red", 320);
  8. Apple apple4 = new Apple("小小", "Green", 300);
  9.  
  10. List<Apple> appleList = Arrays.asList(apple1, apple2, apple3, apple4);
  11.  
  12. //lambda 表达式形式
  13. //appleList.sort((Apple a1, Apple a2) -> {
  14. // return new Double(a1.getWeight() - a2.getWeight()).intValue();
  15. //});
  16.  
  17. //实例方法引用
  18. AppleComparator comparator = new AppleComparator();
  19. appleList.sort(comparator::compareByWeight);
  20.  
  21. appleList.forEach(apple -> System.out.println(apple));
  22.  
  23. }
  24. }
  25. 输出:
  26. Apple{category='红富士', color='Red', weight=280.0}
  27. Apple{category='小小', color='Green', weight=300.0}
  28. Apple{category='哈哈', color='Red', weight=320.0}
  29. Apple{category='冯心', color='Yello', weight=470.0}

通过上面两个例子可以看到,静态方法引用和实例方法引用都是比较好理解的。

(3)类方法引用

一般来说,同类型对象的比较,应该当前调用方法的对象与另外一个对象进行比较,好的设计应该像下面:

  1. public class Apple {
  2.  
  3. private String category;
  4. private String color;
  5. private double weight;
  6.  
  7. public Apple(String category, String color, double weight) {
  8. this.category = category;
  9. this.color = color;
  10. this.weight = weight;
  11. }
  12. //这里和上面静态方式唯一区别就是这个参数就一个,需要实例对象调这个方法
  13. public int compareByWeight(Apple other) {
  14. double diff = this.getWeight() - other.getWeight();
  15. return new Double(diff).intValue();
  16. }
  17.  
  18. //getter setter toString
  19. }

还是之前List排序的例子,看看使用类方法引用如何写:

  1. public class LambdaTest {
  2.  
  3. public static void main(String[] args) {
  4.  
  5. Apple apple1 = new Apple("红富士", "Red", 280);
  6. Apple apple2 = new Apple("黄元帅", "Yello", 470);
  7. Apple apple3 = new Apple("红将军", "Red", 320);
  8. Apple apple4 = new Apple("国光", "Green", 300);
  9.  
  10. List<Apple> appleList = Arrays.asList(apple1, apple2, apple3, apple4);
  11.  
  12. //lambda 表达式形式
  13. //appleList.sort((Apple a1, Apple a2) -> {
  14. // return new Double(a1.getWeight() - a2.getWeight()).intValue();
  15. //});
  16.  
  17. //这里是类方法引用
  18. appleList.sort(Apple::compareByWeight);
  19.  
  20. appleList.forEach(apple -> System.out.println(apple));
  21.  
  22. }
  23. }
  24. 输出:
  25. Apple{category='红富士', color='Red', weight=280.0}
  26. Apple{category='国光', color='Green', weight=300.0}
  27. Apple{category='红将军', color='Red', weight=320.0}
  28. Apple{category='黄元帅', color='Yello', weight=470.0}

这里使用的是:类名::实例方法名。首先要说明的是,方法引用不是方法调用。compareByWeight一定是某个实例调用的,就是lambda表达式的第一个参数,然后lambda

表达式剩下的参数作为 compareByWeight的参数,这样compareByWeight正好符合lambda表达式的定义。

或者也可以这样理解:

(Apple a1, Apple a2) -> { return new Double(a1.getWeight() - a2.getWeight()).intValue(); }

int compareByWeight(Apple other) 需要当前对象调用,然后与另外一个对象比较,并且返回一个int值。可以理解为lambda表达式的第一个参数 a1 赋值给当前对象, 然后 a2

赋值给 other对象,然后返回int值。

(4)构造方法引用

  1. public class ConstructionMethodTest {
  2.  
  3. public String getString(Supplier<String> supplier) {
  4. return supplier.get();
  5. }
  6.  
  7. public static void main(String[] args) {
  8.  
  9. ConstructionMethodTest test = new ConstructionMethodTest();
  10.  
  11. //lambda表达式形式
  12. System.out.println(test.getString(() -> { return new String();}));
  13.  
  14. //构造方法引用形式
  15. System.out.println(test.getString(String::new));
  16.  
  17. }
  18. }

getString 方法接收一个Supplier类型的参数,Supplier 不接收参数,返回一个String。lambda表达式应该这样写:

  1. () -> { return new String();}
    替换成方法引用的形式如下: 实际上调用的是String 无参构造方法
  1. String::new
  2.  

java代码(1)---Java8 Lambda的更多相关文章

  1. Java学习笔记--Java8 Lambda表达式

    Java Lambda表达式入门:http://blog.csdn.net/renfufei/article/details/24600507 lambda内容的介绍:http://swiftlet. ...

  2. JAVA8 Lambda初体验

    import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.u ...

  3. 【Java学习笔记之三十一】详解Java8 lambda表达式

    Java 8 发布日期是2014年3月18日,这次开创性的发布在Java社区引发了不少讨论,并让大家感到激动.特性之一便是随同发布的lambda表达式,它将允许我们将行为传到函数里.在Java 8之前 ...

  4. java代码之美(10)---Java8 Map中的computeIfAbsent方法

    Map中的computeIfAbsent方法 Map接口的实现类如HashMap,ConcurrentHashMap,HashTable等继承了此方法,通过此方法可以在特定需求下,让你的代码更加简洁. ...

  5. java代码之美(14)---Java8 函数式接口

    Java8 函数式接口 之前写了有关JDK8的Lambda表达式:java代码之美(1)---Java8 Lambda 函数式接口可以理解就是为Lambda服务的,它们组合在一起可以让你的代码看去更加 ...

  6. java代码之美(15)---Java8 Function、Consumer、Supplier

    Java8 Function.Consumer.Supplier 有关JDK8新特性之前写了三篇博客: 1.java代码之美(1)---Java8 Lambda 2.java代码之美(2)---Jav ...

  7. java代码(15) ---java8 Function 、Consumer 、Supplier

    Java8 Function.Consumer.Supplier 有关JDK8新特性之前还有三篇博客: 1,java代码(1)---Java8 Lambda 2,java代码(2)---Java8 S ...

  8. java代码(14) --Java8函数式接口

    Java8函数式接口 之前有关JDK8的Lambda表达式 Java代码(1)--Java8 Lambda 函数式接口可以理解就是为Lambda服务的,它们组合在一起可以让你的代码看去更加简洁 一.概 ...

  9. java代码(10) ---Java8 Map中的computeIfAbsent方法

    Map中的computeIfAbsent方法 一.案例说明 1.概述 在JAVA8的Map接口中,增加了一个computeIfAbsent,此方法签名如下: public V computeIfAbs ...

  10. Java8 Lambda表达式和流操作如何让你的代码变慢5倍

    原文出处:ImportNew 有许许多多关于 Java 8 中流效率的讨论,但根据 Alex Zhitnitsky 的测试结果显示:坚持使用传统的 Java 编程风格——iterator 和 for- ...

随机推荐

  1. poi excel自动转换成javabean 支持引用类型属性二级转换

    最近项目需要使用excel导入功能,导入学生的时候需要指定所在班级,使用excel一次性导入! 将以前的代码改改支持属性内引用类的转换. 测试对象为User对象,javabean结构: private ...

  2. xv6 操作系统的环境搭建

    xv6 是 MIT 设计的一个教学型操纵系统.xv6 可在 Intel X86 框架上运行,为了方便,建议将 xv6 运行在 QEMU 虚拟机器上,本人的实验环境是 ubuntu 18.04 . 1. ...

  3. .Net基础之5——复杂数据类型

    (1)复习 1.变量 int  double   string   char   bool    decimal 变量的使用规则:先声明再赋值最后使用 2.Camo      Pascal 3.运算符 ...

  4. 利用Jackson封装常用JsonUtil工具类

    在日常的项目开发中,接口与接口之间.前后端之间的数据传输一般都是使用JSON格式,那必然会封装一些常用的Json数据转化的工具类,本文讲解下如何利用Jackson封装高复用性的Json转换工具类. 转 ...

  5. 如何利用Xpath抓取京东网商品信息

    前几小编分别利用Python正则表达式和BeautifulSoup爬取了京东网商品信息,今天小编利用Xpath来为大家演示一下如何实现京东商品信息的精准匹配~~ HTML文件其实就是由一组尖括号构成的 ...

  6. 阿里P9精心编写高并发设计手册,来看大厂是如何进行系统设计

    在看这篇文章的应该都是IT圈的朋友吧,不知道你们有没有考虑过这样几件事: 淘宝双11的剁手狂欢为什么天猫没崩掉? 为什么滴滴打车高峰如何滴滴依旧可以平稳运行? 为什么疫情期间,钉钉能支撑那么多人同时上 ...

  7. 前端基础进阶(十四):es6常用基础合集

    在实际开发中,ES6已经非常普及了.掌握ES6的知识变成了一种必须.尽管我们在使用时仍然需要经过babel编译. ES6彻底改变了前端的编码风格,可以说对于前端的影响非常巨大.值得高兴的是,如果你熟悉 ...

  8. Python 每日一练(5)

    引言 Python每日一练又开始啦,今天的专题和Excel有关,主要是实现将txt文本中数据写入到Excel中,说来也巧,今天刚好学校要更新各团支部的人员信息,就借此直接把事情做了 主要对于三种数据类 ...

  9. Java中的集合(二)单列集合顶层接口------Collection接口

    Java中的集合(二)单列集合顶层接口------Collection接口 Collection是一个高度封装的集合接口,继承自Iterable接口,它提供了所有集合要实现的默认方法.由于Iterab ...

  10. 非静态代码块(非static初始化块)&静态代码块(static初始化块)

    非静态代码块: TestOrder: package com.aff.singleton; /* 类的第四个成员:初始化块(代码块) 代码块: 如果有修饰的话只能使用static 分类:非静态代码块: ...