《Java8实战》-读书笔记第一章(02)

组合星轨
• 阅读 1039

《Java8实战》-读书笔记第一章(02)

从方法传递到Lambda

接着上次的Predicate,继续来了解一下,如果继续简化代码。

把方法作为值来传递虽然很有用,但是要是有很多类似与isHeavyApple和isGreenApple这种可能只用一两次的方法定义一堆确实有点烦人。为了解决这个问题,Java8它引入了一套新记法(匿名函数或Lambda),然你可以这样写:

List<Apple> isRedApples = filterApples(FilteringApples.apples, apple -> "red".equals(apple.getColor()));

或者是:

List<Apple> appleList = filterApples(FilteringApples.apples, apple -> apple.getWeight() < 120
                && "red".equals(apple.getColor()));

甚至,你都可以不需要使用filterApples这个方法了,直接使用Stream中的filter方法就可以解决了:

List<Apple> isGreenApple = apples.stream().filter(apple -> "green".equals(apple.getColor()))
                .collect(Collectors.toList());

酷,看起来很不错。所以,你甚至都不需要为只用一次的方法写定义;这样的代码看起来更简洁、更清晰,因为你用不着去找自己到底传递了什么代码。

在刚刚筛选苹果的过程中,就有使用到Stream(流)其中的一个方法,这个Stream和InputStream、OutputStream是两个完全不同的东西。Stream它是Java8中的一个核心新特,它是一套新的用来处理集合的API,有很多类似与filter这样的方法而且使用起来非常的简单和简洁,可以简化大部分代码并且在并行的情况下利用多核CPU,能很有效的提升对集合处理的性能。

本章只是简单的介绍了一下流的使用方式,至于流的详细用法后面的章节会提到的。

现在,有一串字符串,需要进行筛选并且转为大写以进行排序,在Java8之前是我们是这么干的:

List<String> stringList = Arrays.asList("a1", "a2", "b1", "c1", "c2", "c4", "c3");

List<String> cList = new ArrayList<>();
for (String s : stringList) {
    // 筛选出以c开头的字符串
    if (s.startsWith("c")) {
        // 将以c开头的字符串转为大写,添加到集合
        cList.add(s.toUpperCase());
    }
}

// 排序
Collections.sort(cList);

// 遍历打印
for (String s : cList) {
    System.out.println(s);
}

这样的代码看起来很头疼,需要写这么长一段的代码,在Java8中可以使用Stream进行优化:

List<String> stringList = Arrays.asList("a1", "a2", "b1", "c1", "c2", "c4", "c3");

stringList.stream()
        // 筛选出以c开头的字符串
        .filter(s -> s.startsWith("c"))
        // 将刚刚以c开头的字符串转为大写
        .map(String::toUpperCase)
        // 排序
        .sorted()
        // 循环遍历
        .forEach(System.out::println);

太棒了,只需要短短的一行代码就可以完成!但是,使用Stream它也是有缺点的,它的性能不如foreach的效率高为了解决这个问题,Stream支持并。使用并行能极大的利用多核CPU的优势,例如说:这些代码原本只是用单核进行处理,现在有一台8核的CPU电脑,那么它的处理速度就会是单核的八倍。

我们来进行比较一下,生成一个0-100的数字并写入到文件中,循序流VS并行流谁的效率更高.

循序流:

long startTime = System.currentTimeMillis();
OutputStream out = new FileOutputStream(new File("D:/integer1.txt"));

IntStream.rangeClosed(0, 100)
        .forEach(i -> {
            try {
                Thread.sleep(100L);
                out.write(i);
            } catch (IOException | InterruptedException e) {
                e.printStackTrace();
            }
        });

long endTime = System.currentTimeMillis();
System.out.println("循序流:" + (endTime - startTime));

并行流:

long startTime = System.currentTimeMillis();
OutputStream out = new FileOutputStream(new File("D:/integer2.txt"));

IntStream.rangeClosed(0, 100)
        .parallel().forEach(i -> {
    try {
        Thread.sleep(100L);
        out.write(i);
    } catch (IOException | InterruptedException e) {
        e.printStackTrace();
    }
});

long endTime = System.currentTimeMillis();
System.out.println("并行流:" + (endTime - startTime));

执行结果(I5-6200U的笔记本上执行结果):

循序流:10251
并行流:2620

效率明显要比循序流快很多嘛!但是,并行流并不是万能的,如果把sleep去掉后并且数字加到100万,你会发现运行的时间比循序流还要长。

去掉sleep并且生成的数字是0-100万,所消耗的时间:

循序流:2775
并行流:3346

至于为什么有时候并行流效率比循序流还低,这个以后的文章会解释。

默认方法

默认方法是Java8中的一个新特性,它的出现使得接口的升级变得平滑了,因为子类不是必须再去显式的实现接口中的方法了。

例如:在Java8中,你可以直接调用List接口中的sort方法、它是用Java8 List接口中如下所示的默认方法实现的:

default void sort(Comparator<? super E> c) {
    Object[] a = this.toArray();
    Arrays.sort(a, (Comparator) c);
    ListIterator<E> i = this.listIterator();
    for (Object e : a) {
        i.next();
        i.set((E) e);
    }
}

这意味着List的任何实体类都不需要显式的实现sort,而在以前的Java版本中,除非提供了sort的实现,否则这些实体类都无法编译通过。但是,默认方法也存在着一些问题,一个类可以实现多个接口,那么好几个接口多有同样的默认方法,那么这是否意味着Java中有了某种形式的多继承?如果是多继承,那么会不会出现像C++中菱形继承的问题?这些问题以后的文章中都会有解释和解决方案。

第一章总结:

  1. 了解了Java8中的一些核心新特性,例如:Lambda表达式、Stream、默认方法。
  2. 了解了Lambda表达式和Stream为代码带来的简洁性。
  3. 并行流带来的好处。
  4. Java8中的默认方法带来的好处。

代码案例:

chap1

点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
3年前
Java日期时间API系列31
  时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总毫秒数,是所有时间的基础,其他时间可以通过时间戳转换得到。Java中本来已经有相关获取时间戳的方法,Java8后增加新的类Instant等专用于处理时间戳问题。 1获取时间戳的方法和性能对比1.1获取时间戳方法Java8以前
Wesley13 Wesley13
3年前
java8实战读书笔记:复合Lambda表达式
本节将重点探讨复合Lambda表达式的使用。在阅读本篇之前建议您先阅读:java8实战读书笔记:Lambda表达式语法与函数式编程接口,因为本篇是上篇的补充。(https://www.oschina.net/action/GoToLink?urlhttp%3A%2F%2Fmp.weixin.qq.com%2Fs%3F__biz%3DMzIzNzgy
Wesley13 Wesley13
3年前
java8读书笔记:探究java8流收集数据原理
java8专栏目录:1.java8实战读书笔记:Lambda表达式语法与函数式编程接口(https://www.oschina.net/action/GoToLink?urlhttp%3A%2F%2Fmp.weixin.qq.com%2Fs%3F__biz%3DMzIzNzgyMjYxOQ%3D%3D%26mid%3D2247484246%
阿邹 阿邹
4年前
java传值和传引用问题
这个问题还是很常见的,如果你平常敲代码比较多你可能经常会遇到这个问题。如果你知道java这个机制,你可能还会一直在找代码的问题。java中的值传递和引用传递。比如下面有这俩个方法javaprivatevoidupdataValue(Strings){s"123";}privatevoidupd
Stella981 Stella981
3年前
Google Map 开发(一):中获取兴趣点 (POI) 信息
首先来说,GoogleMap的集成其实非常简单,如果使用了GooglePlayService了可能就几行代码调用的事,不过在我集成的过程中,发现虽然基于ServiceSDK的集成虽然简单,但是功能少了很多,比如基于POI信息获取的方法函数中,就没有传递半径区域的方法,这就需要使用到PlacesAPIWebService了。
Wesley13 Wesley13
3年前
Java8特性
Java8又称jdk1.8。主要新特性:Lambda表达式 −Lambda允许把函数作为一个方法的参数(函数作为参数传递进方法中。方法引用 −方法引用提供了非常有用的语法,可以直接引用已有Java类或对象(实例)的方法或构造器。与lambda联合使用,方法引用可以使语言的构造更紧凑简洁,减少冗余代码。
Wesley13 Wesley13
3年前
Java8新特性学习
1简述公司自年初终于开始使用java8作为项目的标准jdk,在开发过程中,逐渐认识到java8的很多新特性,确实很方便.其中内容的核心,在于函数式编程,即将函数本身作为对象参数去处理.其中涉及到三个关键新特性:1.lambda表达式(及函数式接口)2.stream3.方法引用这三个新特性的使用是相辅相
Wesley13 Wesley13
3年前
Java8的这些集合骚操作,你掌握了嘛?
Java8时Lambda表达式的出现,将行为作为参数传递进函数的函数式编程,大大简化了之前冗杂的写法。对于集合一类,我们来整理一下发生的变化吧。!Java8的这些集合骚操作,你掌握了嘛?(https://p6tt.byteimg.com/origin/dficimagehandler/e5ad919fe84f4ae7b7c8395f5
Wesley13 Wesley13
3年前
JAVA初学笔记&宋红康JAVA基础、高级篇(其九)
关于类与方法的使用技巧方法重载定义:同一个类中,允许多个同名方法,只需所传递的参数类型不同即可(类似于路由系统)使用:根据传递的类型自动区分到对应的方法值传递多个实参传递JDK5.0之前:publicstaticvoidtest(inta,Str
Wesley13 Wesley13
3年前
Java日期时间API系列30
  实际使用中,经常需要使用不同精确度的Date,比如保留到天2020042300:00:00,保留到小时,保留到分钟,保留到秒等,常见的方法是通过格式化到指定精确度(比如:yyyyMMdd),然后再解析为Date。Java8中可以用更多的方法来实现这个需求,下面使用三种方法:使用Format方法、 使用Of方法和使用With方法,性能对比,使用
Wesley13 Wesley13
3年前
JAVA 数组作为方法返回值—返回地址
packageCode411;/一个方法可以有0,1,多个参数,但只能有0和1个返回值希望一个方法产生多个结果数据进行返回数组作为方法的参数,传递进去的是数组的地址值。/publicclassCodeAaaryReturn{publicstaticvoidmain(Stringargs){int
组合星轨
组合星轨
Lv1
万里归船弄长笛,此心吾与白鸥盟。
文章
4
粉丝
0
获赞
0