《Java 8实战》学习总结

Java 8新特性概览

  1. Lambda表达式
  2. 默认方法

Lambda表达式

  1. Lambda和函数式接口

    Java 8中新增了函数 -- 值的一种新形式. 在运行时传递方法/函数能将方法/函数变成"一等值"(传统编程语言中, 值(eg: 实例)是"一等公民", 值的结构(eg: 类和方法)是"二等公民").
    Lambda是表示函数的强有力的方式. 它允许你直接以内联的形式为函数式接口的抽象方法提供实现, 并把整个表达式作为函数式接口的实例.

    • 函数式接口: 只定义一个抽象方法的接口.

        @FunctionalInterface
        public interface Predicate<T> {
        
            boolean test(T t);
        }
      
    • Lambda的签名: 函数式接口的抽象方法的签名基本上就是Lambda表达式的签名 (只不过Lambda没有方法名).

        Predicate<T> ((T) -> boolean)
      
  2. Lambda与方法引用 (从现有方法获取函数)

    • 静态方法引用: [className]::[method]

    • 实例方法引用: [className]::[method]

      注意, 实例方法引用和静态方法引用格式相同; 区别是, 实例方法引用需要以每个元素为引用调用指定的方法

      list.stream().map(Objects::isNull  /* 静态方法引用, 每个元素作为toString的参数, 元素本身不需要有该方法 */)
      list.stream().map(Object::toString /* 实例方法引用, 每个元素作为toString的调用对象 */)
      
    • 已有对象的方法引用: [objName]::[method]

  3. Java 8自带的常用函数式接口(java.util.function包)

    • Predicate<T>, Consumer<T>, Function<F, T>, Supplier<T>, UnaryOperator<T>
    • BiPredicate<L, R>, BiConsumer<T, U>, BiFunction<T, U, R>, BinaryOperator<T>
    • 特化的函数: IntFunction<R>, LongFunction<R>, DoubleFunction<R>

  1. 流和集合

      • 从支持数据处理操作的源生成的一系列元素
    • 流与集合

      • 一次性迭代: 只能遍历一次
      • 内部迭代:环绕执行模式

      Collection主要是为了存储和访问数据, 而Stream则主要用于描述对数据的计算.
      与集合不同, 流并没有把所有元素保存在内存中, 而是实时计算需要的元素, 对同一个流而言这个过程是不可重置的, 每次迭代都必须生成新的流;
      流的迭代是内置的, 对流的使用往往一气呵成 -- 我们只管组装/转换流, 通过函数定制流的行为, 并最终消费流, 其中对每一个元素的命令式迭代不需要我们来关心.

    • 流的操作类型

      • 中间操作: 给流附加转换/计算规则, 将流包装成另一个流 (可以把流组装成包含复杂操作的流水线)
      • 终端操作: 驱动流的计算, 得到结果

      要完成一次流式计算, 一个流要经过"零个或多个中间操作"将自己组装成流水线, 然后由"一个终端操作"最终驱动流水线的运行;
      流是慵懒的, 这意味着, 只有调用了终端操作, 才会发生实际计算.

  2. 流的基本使用

    • 流的基本操作 (Stream API)
      • 筛选和切片: filter, limit, skip
      • 映射: map, flatMap
      • 查找和匹配: anyMatch, allMatch, noneMatch, findFirst, findAny
      • 归约: reduce
    • 数值流
      • 原始类型特化:IntStream, LongStream, DoubleStream
    • 流的构建
      • 集合: Collection.stream
      • 值: Stream.of
      • 数组: Arrays.stream
      • 文件: Files.lines
      • 无限流: Stream.iterate, Stream.generate
    • 流的调试
      • 查看流的运行轨迹: Stream.peek
  3. 流和收集器

    • 流的收集
      • 归约和汇总: ...
      • 分组(多级分组): Collectors.groupingBy
      • 分区(多级分区): Collectors.partitioningBy
    • 收集器:
      • Collector接口

          public interface Collector<T, A, R> {
              // A function that creates and returns a new mutable result container.
              Supplier<A> supplier();
          
              // A function that folds a value into a mutable result container.
              BiConsumer<A, T> accumulator();
          
              // A function that accepts two partial results and merges them.
              BinaryOperator<A> combiner();
          
              // Perform the final transformation from the intermediate accumulation type "A" to the final result type "R".
              Function<A, R> finisher();
          
              // Returns a Set of "Collector.Characteristics" indicating the characteristics of this Collector.
              // This set should be immutable.
              Set<Characteristics> characteristics();
          }
        
      • 自定义收集

        参见Collectors工具类中的默认实现, eg:

        toList(), toSet(), toCollection()
        counting(), maxBy()
        groupingBy(), reducing(), joining()

  4. 流的并行化

    • 并行流:
      • 顺序流并行化
      • 并行化的考虑(正确并行,高效并行)
    • 分支合并框架:
      • 分支合并框架的原理和实践
      • 工作窃取
    • 新型可拆分的迭代器:
      • Spliterator接口

          public interface Spliterator<T> {
              // If a remaining element exists, performs the given action on it, returning true; else returns false.
              boolean tryAdvance(Consumer<? super T> action);
              
              // If this spliterator can be partitioned, returns a Spliterator covering elements, that will, 
              // upon return from this method, not be covered by this Spliterator.
              Spliterator<T> trySplit();
        
              // Returns an estimate of the number of elements that would be encountered by a #forEachRemaining traversal, 
              // or returns Long#MAX_VALUE if infinite, unknown, or too expensive to compute.
              long estimateSize();
              
              // Returns a set of characteristics of this Spliterator and its elements. 
              int characteristics();
          }
        
      • Stream/Spliterator vs Iterable/Iterator

默认方法

// TODO

Java 8与响应式编程

  1. 组合式异步编程:Future -> ListenableFuture -> CompletableFuture

  2. 异步化和异常处理

  3. 异步任务和组合/连接/...

    • Future/RunnableFuture

      • 异步任务的提交/执行: ExecutorService.submit/execute

      • 判断完成状态: isDone, isCancelled

      • 进入完成状态: run, cancel

      • 获取完成结果: get

          // 假设Math.random方法是一个耗时较长的方法, 需要使用异步计算
          Future<Double> future = executorService.submit(Math::random);
          
          // 执行到需要依赖future的返回结果的位置, 需要获取结果, 可能导致阻塞
          // 可能需要预先判断(isDone, isCancelled) && 需要捕获异常(ExecutionException, InterruptedException)
          Double result = future.get();
        
    • ListenableFuture (extends Future)

      • 添加异步回调: addListener

          // 将普通线程池装饰成ListeningExecutorService, 从而可以返回ListenableFuture
          ListeningExecutorService listeningExecutorService = MoreExecutors.listeningDecorator(executorService);
        
          // 假设Math.random方法是一个耗时较长的方法, 需要使用异步计算
          ListenableFuture<Double> listenableFuture = listeningExecutorService.submit(Math::random);
        
          // 假设有后续操作需要依赖listenableFuture的返回结果, 可以以回调的方式异步处理, 避免阻塞
          // Futures.addCallback封装了异常处理, 并且回调的方式避免了对任务状态的判断
          // 注意, 这里回调的执行使用了同一个线程池; 如果不指定线程池, 则默认在原任务线程中执行回调
          Futures.addCallback(listenableFuture, new FutureCallback<Double>() {
              @Override
              public void onSuccess(Double aDouble) {
                  // on success
              }
        
              @Override
              public void onFailure(Throwable throwable) {
                  // on failure
              }
          }, executorService);
        

        可以看到, 异步API的返回/响应方式, 要么是通过回调函数, 要么是由调用方再次执行一个"等待, 直到计算完成"的方法调用.

    • CompletableFuture (implements Future, CompletionStage)

      Future接口的局限性: 难以表述Future结果之间的依赖性.
      我们需要更具描述能力的特性, 考虑以下场景:

      • 将两个异步计算合并为一个
      • 等待Future集合中的所有任务都完成
      • 仅等待Future集合中最快结束的任务完成
      • 以手工设定异步操作结果的方式完成一个Future任务的执行
      • 应对Future的完成事件

      CompletableFuture和Stream的设计都遵循了类似的模式: 它们都使用了Lambda表达式以及流水线的思想.
      从这个角度, 你可以说CompletableFuture和Future的关系就跟Stream和Collection的关系一样.

      • 同步API转异步化API

          // 假设getPrice方法是一个耗时较长的方法, 需要改成异步API
          public Double getPrice() {
              return Math.random() * 10000;
          }
        
          // 1.1
          public Future<Double> getPriceAsync() {
              CompletableFuture<Double> completableFuture = new CompletableFuture<>();
              new Thread(() -> {
                  try {
                      double price = getPrice();
                      completableFuture.complete(price); // 通过设置返回值的方式完成Future (由执行任务的线程)
                  } catch (Exception e) {
                      completableFuture.completeExceptionally(e); // 记录异常, 以便在Future.get中重新抛出
                  }
              }).start();
        
              return completableFuture;
          }
        
          // 1.2
          public Future<Double> getPriceAsyncV2() {
              return CompletableFuture.supplyAsync(this::getPrice); // 通过CompletableFuture的supplyAsync工厂方法
          }
        
      • 并行流 vs CompletableFuture流

          // 2.1
          public List<Double> getPrices() {
              // 模拟100次请求
              IntStream intStream = IntStream.iterate(1, i -> i + 1).limit(100);
              // 并行化
              Stream<Double> doubleStream = intStream.parallel().mapToObj(i -> getPrice());
              return doubleStream.collect(Collectors.toList());
          }
        
          // 2.2
          public List<Double> getPricesAsync() {
              // 模拟100次请求
              IntStream intStream = IntStream.iterate(1, i -> i + 1).limit(100);
              // 异步化; 需要调用collect触发所有异步任务并返回future引用
              List<CompletableFuture<Double>> futures = intStream.mapToObj(i -> CompletableFuture.supplyAsync(this::getPrice))
                      .collect(Collectors.toList());
              // 获取结果; CompletableFuture.join类似get, 区别是不会抛出任何受检异常 (适合用作函数进行传递)
              return futures.stream().map(CompletableFuture::join).collect(Collectors.toList());
          }
        

        CompletableFuture的优势:

        1. 灵活的线程池配置 (N[threads] = N[cpu] * U[cpu] * (1 + W/C))
        2. 对顺序流(不易并行的)流有同样好的效果
        3. 无论是IO密集型还是CPU密集型都同样适用
      • 组合式异步 (CompletableFuture + Stream, 同步 + 异步)

          // 假设getDiscountedPrice方法是另一个耗时较长的方法, 需要用于和getPrice方法的结果组合生成折扣价
          private Double getDiscountedPrice(Double price) {
              return price * 0.88d;
          }
          
          // 3.
          public List<Double> getDiscountedPricesAsync() {
              // 模拟100次请求
              IntStream intStream = IntStream.iterate(1, i -> i + 1).limit(100);
              List<CompletableFuture<Double>> futures = intStream.mapToObj(i -> CompletableFuture.supplyAsync(this::getPrice))
                      // 组合同步调用: 打印future的初步结果
                      .map(future -> future.thenApply(this::print))
                      // 组合异步调用: 计算折扣价
                      // 这里虽然调用的是thenCompose而不是thenComposeAsync, 但是使用了CompletableFuture.supplyAsync, 因此仍是异步的
                      .map(future -> future.thenCompose(r2 -> CompletableFuture.supplyAsync(() -> getDiscountedPrice(r2))))
                      .collect(Collectors.toList()); // 注意future之间的同步/异步依赖关系
        
              return futures.stream().map(CompletableFuture::join).collect(Collectors.toList());
          }
        
          private Double print(Double d) {
              System.out.println(d);
              return d;
          }
        

        通常而言, 同步的调用和它的前一个任务一样, 在同一个线程中运行(future.isDone()的情况下会由调用线程直接执行?!); 而异步的调用会将后续的任务提交到一个线程池, 由不同的线程处理.
        可以类比带Executor参数和不带Executor参数的Futures.addCallback方法.

      • 来自CompletionStage接口的方法 (下列的每个方法几乎都有个对应的异步版本, 方法名带Async后缀):

        • thenApply, thenAccept, thenRun, thenCombine, thenCompose

        • thenAcceptBoth, runAfterBoth

        • applyToEither, acceptEither, runAfterEither

        • exceptionally, whenComplete, handle

        • toCompletableFuture

          // TODO

Java 8与函数式编程

// TODO

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

推荐阅读更多精彩内容

  • Java8 in action 没有共享的可变数据,将方法和函数即代码传递给其他方法的能力就是我们平常所说的函数式...
    铁牛很铁阅读 1,227评论 1 2
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139
  • 第一章 为什么要关心Java 8 使用Stream库来选择最佳低级执行机制可以避免使用Synchronized(同...
    谢随安阅读 1,490评论 0 4
  • 《拒》 风催舟荡一江月; 桨拒水棹满溪云。
    自命飞皇Yoes阅读 265评论 0 1
  • 第73届教育装备展今天在广州市海珠区琶洲国际会展中心开幕,来自全国各地的众多商家和参观者参加了开幕式。 简短热烈的...
    卓姿宝坐姿好阅读 195评论 0 0