Java8特性之Lambda、方法引用和Streams

Java8特性之Lambda、方法引用和Streams

Java8已经推出了好一段时间了,而掌握Java8的新特性也是必要的,如果要进行Spring开发,那么可以发现 Spring的官网已经全部使用Java8来编写示例代码了,所以,不学就看不懂。

这里涉及三个重要特性:

  • Lambda
  • 方法引用
  • Streams

Lambda

最早了解Lambda是在C#中,而从Java8开始,Lambda也成为了新的特性,而这个新的特性的目的,就是为 了消除单方法接口实现的匿名内部类

在Java8以前的版本中,定义一个Thread是这样的:

final int i = 0;
new Thread(new Runnable() {
     @Override
     public void run() {
         System.out.println("i = " + i);
     }
}).start();

而Lambda是这样的:

int i = 0;
new Thread(() -> System.out.println("i = " + i));

首先不看Lambda本身的写法,可以发现,对于i值 的访问,在Lambda中已经不需要声明i为final了。

其次,要明白一个重要的道理:Lambda 要求实现的接口中只有一个方法,像上面的Runnable接口就只有一个 run方法,如果一个接口中有多于一个方法,则不能写成Lambda的形式。

最后来看标准的Lambda表达式的结构:

包含三个部分

  1. 一个括号内用逗号分隔的形式参数,参数是函数式接口里面方法的参数
  2. 一个箭头符号:->
  3. 方法体,可以是表达式和代码块,方法体函数式接口里面方法的实现,如果是代码块,则必须用{}来包裹起来,且需要一个return 返回值,但有个例外,若函数式接口里面方法返回值是void,则无需{}

总体看起来像这样

(parameters) -> expression 或者 (parameters) -> { statements; }

结构很简单,小括号表示参数列表,大括号表示方法体,中间使用一个 "->" 隔开即可。

这里的参数体和方法体分别指的是接口中方法的参数体和方法体

接着我们看一个比较比较复杂的:

ArrayList<Integer> integers = new ArrayList<>();
integers.add(6);
integers.add(2);
integers.add(5);
integers.sort((o1, o2) -> o1 - o2);
System.out.println(integers);

可以看到,创建了一个ArrayList,里面的元素是Integer类型,接着调用sort()方法对其进行排 序,sort方法接受一个Comparator接口的实现类,这个接口中有且仅有一个方法compare,所以如果使用 匿名内部类的写法,如下所示:

integers.sort(new Comparator<Integer>() {
     @Override
     public int compare(Integer o1, Integer o2) {
         return o1 - o2;
     }
});

可以发现,这里参数列表和方法体都很明白了,要注意的是,这里的方法体中不带{},原因就是当 方法体只有一个语句的时候,{}可以省略

另外,return关键字也被省略 了,原因是编译器会认为,既然只有一个语句,那么这个语句执行的结果就应该是返回值,所以return也就不需 要了。同理,当参数只有一个的时候,小括号也是可以省略的

明白这种对应的关系,Lambda就算是掌握了。

方法引用

方法引用包括几种情况:

  • 静态方法引用
  • 构造方法引用
  • 类成员方法引用
  • 对象方法引用

要注意这里的方法引用实际上是某些Lambda表达式 的更简洁写法,原因就是在这些情况下,编译器能够智能的推断出参数体中的值究竟是方法的传入参数还是调用者。

先定义一个Car类:

  import java.util.function.Supplier;
  
  public class Car {
      // 通过Supplier获取Car实例
      public static Car create(Supplier<Car> supplier) {
          return supplier.get();
      }
  
      // 静态方法,一个入参Car对象
     public static void collide(final Car car) {
         System.out.println("Collide " + car.toString());
     }
 
     // 一个入参Car
     public void follow(final Car car) {
         System.out.println("Following car " + car.toString());
     }
 
     // 不带入参
     public void repair() {
         System.out.println("Repaired car " + this.toString());
     }
 }

构造方法引用

Supplier接口的定义:

@FunctionalInterface
public interface Supplier<T> {

    /**
     * Gets a result.
     *
     * @return a result
     */
    T get();
}

这个接口被FunctionalInterface注解声明了,这个注解是一个新的注解,表明这个接口是一个函数式接口,只有一个抽象方法。

函数式接口(functional interface 也叫功能性接口,其实是同一个东西)。简单来说,函数式接口是只包含一个方法的接口。比如Java标准库中的java.lang.Runnable和java.util.Comparator都是典型的函数式接口。java 8提供 @FunctionalInterface作为注解,这个注解是非必须的,只要接口符合函数式接口的标准(即只包含一个方法的接口),虚拟机会自动判断,但 最好在接口上使用注解@FunctionalInterface进行声明,以免团队的其他人员错误地往接口中添加新的方法。

所以我们调用静态方法create创建Car对象的时候,代码应该是如下所示的:

 Car.create(new Supplier<Car>() {
     @Override
     public Car get() {
         return new Car();
     }
 });

但是我们才说了Lambda表达式,所以更简单的写法就是:

 Car.create(()->new Car());

可以看到,Car类存在一个不带参数的构造方法,所以编译器不需要根据参数列表猜测构造方法的参数(因为都是空的),所以就有一个更加简单的写法:

 Car.create(Car::new);

实际上,如果Lambda的参数个数和类的构造方法个数一致,也可以改写为上面的形式,只要是没有歧义即可。

静态方法引用

这里开始会涉及一些Streams内容,但是可以先忽略,后面会详细说。

我们创建一个Car对象,接着将其添加进一个List中:

final Car car = Car.create(Car::new);
final List<Car> cars = Arrays.asList(car); 

Java8中给Iterable接口添加了forEach方法方便我们遍历集合类型。

现在假设我们要给List中的每个Car对象调用一次Car.collide(Car car)静态方法,那么可以使用forEach方法,而forEach方法需要传入一个Consumer,恰好,这个Consumer接口也带有 FunctionalInterface注解,所以我们一步一步的来看:

 cars.forEach(new Consumer<Car>() {
     @Override
     public void accept(Car car) {
         Car.collide(car);
     }
 });

写成Lambda:

 cars.forEach(c -> Car.collide(c));

就是对传进来的Car对象执行静态方法,很简单。但是实际上,对于静态方法,编译器也不需要推断调用者(类名),当传入参数和静态方法所需参数个数一致时,就不存在歧义:

所以这里可以直接使用方法引用:

 cars.forEach(Car::collide);

类成员方法引用

类的成员方法不能是静态的,而这个情况其实和静态方法类似,区别是,Lambda表达式的参数个数需要等于所调用方法的入参个数加一。

为什么要加一?

因为类的成员方法不能通过类名直接调用,只能通过对象来调用,也就是Lambda 表达式的第一个参数,是方法的调用者,从第二个开始的参数个数要和需要调用方法的入参个数一致即可。

对于上面的例子,如果要对List中的每个对象执行一次它的repair方法:

 cars.forEach(c -> c.repair());

根据上图,这里参数只有一个,而repair方法没有入参,所以不存在歧义,即可以改写为对应的方法引用:

 cars.forEach(Car::repair);

对象方法引用

与类方法引用不同的是,对象方法引用 方法的调用者是一个外部的对象

对于上面例子,可以再创建一个Car的对象police,并让police调用follow方法跟踪List中的每个 Car:

final Car police = Car.create(Car::new);
cars.forEach((car1) -> police.follow(car1));

改成对象方法引用:

 cars.forEach(police::follow);

至此,方法引用也完成了。

Streams

Streams的思想很简单,就是遍历。

一个流的生命周期分为三个阶段:

  1. 生成
  2. 操作、变换(可以多次)
  3. 消耗(只有一次)

生成

生成Stream对象

 // . 对象
Stream stream = Stream.of("a", "b", "c");
 // 2. 数组
String [] strArray = new String[] {"a", "b", "c"};
stream = Stream.of(strArray);
stream = Arrays.stream(strArray);
 // 3. 集合
List<String> list = Arrays.asList(strArray);
stream = list.stream();

生成DoubleSteam、IntSteram或LongStream对象(这是目前支持的三个数值类型 Stream对象)

IntStream.of(new int[]{, 2, 3}); // 根据数组生成
IntStream.range(, 3); // 按照范围生成,不包括3
IntStream.rangeClosed(, 3); // 按照范围生成,包括3

等等。。。

变换

一个流可以经过多次的变换,变换的结果仍然是一个流。

常见的变换:map (mapToInt, flatMap 等)、 filter、 distinct、 sorted、 peek、 limit、 skip、 parallel、 sequential、 unordered

消耗

一个流对应一个消耗操作。

常见的消耗操作:forEach、 forEachOrdered、 toArray、 reduce、 collect、 min、 max、 count、 anyMatch、 allMatch、 noneMatch、 findFirst、 findAny、 iterator

例子:

定义一个学生类:

  public class Student {
      public enum Sax{
          FEMALE, MALE
      }
  
      private String name;
      private int age;
      private Sax sax;
      private int height;
 
     public Student(String name, int age, Sax sax, int height) {
         this.name = name;
         this.age = age;
         this.height = height;
         this.sax = sax;
     }
 
     public String getName() {
         return name;
     }
 
     public int getAge() {
         return age;
     }
 
     public Sax getSax() {
         return sax;
     }
 
     public int getHeight() {
         return height;
     }
 
     @Override
     public String toString() {
         return "Student{" +
                 "name='" + name + '\'' +
                 ", age=" + age +
                 ", sax=" + sax +
                 ", height=" + height +
                 '}';
     }
 }

在main方法中创建示例数据:

List<Student> students = Arrays.asList(
         new Student("Fndroid", 22, Student.Sax.MALE, 80),
         new Student("Jack", 20, Student.Sax.MALE, 70),
         new Student("Liliy", 8, Student.Sax.FEMALE, 60)
);

下面实现几个查询:

输出所有性别为MALE的学生:

循环:

for (Student student : students) {
     if (student.getSax() == Student.Sax.MALE) {
         System.out.println(student);
     }
}

使用Stream:

students.stream() // 打开流
         .filter(student -> student.getSax() == Student.Sax.MALE) // 进行过滤
         .forEach(System.out::println); // 输出

求出所有学生的平均年龄:

OptionalDouble averageAge = students.stream()
         .mapToInt(Student::getAge) // 将对象映射为整型
         .average(); // 根据整形数据求平均值
 System.out.println("所有学生的平均年龄为:" + averageAge.orElse(0));

可以看到这里的average方法得到一个OptionalDouble类型的值,这也是Java8的新增特性,OptionalXXX类用于减少空指针异常带来的崩溃,可以通过orElse方法获得其值,如果值为null,则取默认值0。

输出每个学生姓名的大写形式:

List<String> names = students.stream()
         .map(Student::getName) // 将Student对象映射为String(姓名)
         .map(String::toUpperCase) // 将姓名转为小写
         .collect(Collectors.toList()); // 生成列表
System.out.println("所有学生姓名的大写为:" + names);

按照年龄从小到大排序:

List<Student> sortedStudents = students.stream()
         .sorted((o1, o2) -> o1.getAge() - o2.getAge()) // 按照年龄排序
         .collect(Collectors.toList()); // 生成列表
System.out.println("按年龄排序后列表为:" + sortedStudents);

5.判断是否存在名为Fndroid的学生:

boolean isContain = students.stream()
        .anyMatch(student -> student.getName().equals("Fndroid")); // 查询任意匹配项是否存在
System.out.println("是否包含姓名为Fndroid的学生:" + isContain);

将所有学生按照性别分组:

 Map<Student.Sax, List<Student>> groupBySax =         students.stream()
         .collect(Collectors.groupingBy(Student::getSax)); // 根据性别进行分组
 System.out.println(groupBySax.get(Student.Sax.FEMALE));

求出每个学生身高比例:

double sumHeight = students.stream().mapToInt(Student::getHeight).sum(); // 求出身高总和
DecimalFormat formator = new DecimalFormat("##.00"); // 保留两位小数
List<String> percentages = students.stream()
         .mapToInt(Student::getHeight) // 将Student对象映射为身高整型值
         .mapToDouble(value -> value / sumHeight * 00) // 求出比例
         .mapToObj(per -> formator.format(per) + "%") // 组装为字符串
         .collect(Collectors.toList()); 
System.out.println("所有学生身高比例:" + percentages);

欢迎转载,转载请注明出处!
独立域名博客:flywill.cn
欢迎关注公众微信号:Java小镇V
分享自己的学习 & 学习资料 & 生活
想要交流的朋友也可以加微信号备注入群:EscUpDn

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容

  • Java 8自Java 5(发行于2004)以来最具革命性的版本。Java 8 为Java语言、编译器、类库、开发...
    huoyl0410阅读 625评论 1 2
  • Java 8自Java 5(发行于2004)以来最具革命性的版本。Java 8 为Java语言、编译器、类库、开发...
    谁在烽烟彼岸阅读 888评论 0 4
  • 目录结构 介绍 Java语言的新特性2.1 Lambdas表达式与Functional接口2.2 接口的默认与静态...
    夜风月圆阅读 473评论 0 2
  • 前言:Java 8 已经发布很久了,很多报道表明Java 8 是一次重大的版本升级。在Java Code Geek...
    糖宝_阅读 1,322评论 1 1
  • 简介 概念 Lambda 表达式可以理解为简洁地表示可传递的匿名函数的一种方式:它没有名称,但它有参数列表、函数主...
    刘涤生阅读 3,201评论 5 18