美文网首页
Java8 学习总结 - 基本思想

Java8 学习总结 - 基本思想

作者: little田同学 | 来源:发表于2017-10-05 11:08 被阅读0次

    1. 行为参数化

    在java8之前, 我们想给方法传递不同的行为, 最好的办法就是匿名类了:
    比如下面我们想从一个装满苹果的list中的筛选出红苹果,需要传入一个 ApplePredicate, 于是我们在调用filterApples会传入一个匿名类, 实现了test方法来判断苹果的颜色是否为红色。

    //过滤方法
    public static List<Apple> filterApples(List<Apple> inventory, ApplePredicate p) {
      List<Apple> result = new ArrayList<>();
      for (Apple apple : inventory){
        if (p.test(apple)){
          result.add(apple);
        }
      }
      return result;
    }
    
    //接口
    public interface ApplePredicate{
      boolean test (Apple apple);
    }
    
    //方法调用
    List<Apple> redApples = filterApples(inventory, new ApplePredicate() { 
      public boolean test(Apple a){
        return "red".equals(a.getColor());
      }
    });
    

    其实我们需要的就是不同的test方法,但是由于在java8之前不支持Lambda表达式, 所以也需要创建一个对象来实现这个方法。
    但是java8我们可以这么来完成这件事:

    List<Apple> result =
            filterApples(inventory, (Apple apple) -> "red".equals(apple.getColor()));
    

    (Apple apple) -> "red".equals(apple.getColor()) 便是在java8中可以直接传递的函数, 这种表达式叫做Lamda表达式,可能现在还看不懂这个表达, 下面我们会详细讲解下lambda表达式。

    lamda表达式结构
    • 参数列表——这里它采用了 Comparator 中 compare 方法的参数,两个 Apple 。
    • 箭头——箭头 -> 把参数列表与Lambda主体分隔开。
    • Lambda主体——比较两个 Apple 的重量。表达式就是Lambda的返回值了。

    在参数列表里的参数可以在后面的Lambda表达式中使用, 而后面表达式执行结束的值是默认的返回值, 也可以使用return来显式返回。

    2. Lambda表达式在哪里使用?

    什么条件下Lambda表达式可以作为参数传递呢? 我们需要函数式接口来作为参数。
    之前我们在筛选苹果时使用了接口:

    public interface ApplePredicate{
      boolean test (Apple apple);
    }
    

    这样只定义了一个抽象方法的接口便是函数式接口。 上面我们使用Lambda表达式或者使用匿名类都是对这个接口进行了实现, 原则上是一样的。

    3. 函数描述符:

    函数式接口的抽象方法的签名基本上就是Lambda表达式的签名。我们将这种抽象方法叫作
    函数描述符
    例如上面ApplePredicate的函数描述符就是 Apple -> boolean, 传入一个Apple对象返回一个boolean。

    另外java提供了一个标注: @FunctionalInterface
    这个标注用于表示该接口会设计成一个函数式接口。如果你用 @FunctionalInterface 定义了一个接口,而它却不是函数式接口的话,编译器将返回一个提示原因的错误。

    使用函数式接口

    Java8提供了几个函数式接口, 下面来分别介绍下:

    Predicate
    @FunctionalInterface
    public interface Predicate<T>{
      boolean test(T t);
    }
    
    
    public static <T> List<T> filter(List<T> list, Predicate<T> p) {
      List<T> results = new ArrayList<>();
      for(T s: list){
        if(p.test(s)){
          results.add(s);
        }
      }
      return results;
    }
    Predicate<String> nonEmptyStringPredicate = (String s) -> !s.isEmpty();
    List<String> nonEmpty = filter(listOfStrings, nonEmptyStringPredicate);
    

    Predicate定义了一个叫test的抽象方法, 接受泛型T对象, 并返回一个boolean, 可以在过滤时使用

    Consumer
    @FunctionalInterface
    public interface Consumer<T>{
      void accept(T t);
    }
    
    
    public static <T> void forEach(List<T> list, Consumer<T> c){
      for(T i: list){
        c.accept(i);
      }
    }
    forEach(Arrays.asList(1,2,3,4,5), (Integer i) -> System.out.println(i));
    

    Consumer定义了一个accept的抽象方法,接受泛型T对象,不返回

    Function
    @FunctionalInterface
    public interface Function<T, R>{
      R apply(T t);
    }
    
    public static <T, R> List<R> map(List<T> list, Function<T, R> f) {
      List<R> result = new ArrayList<>();
      for(T s: list){
        result.add(f.apply(s));
      }
      return result;
    }
    List<Integer> l = map(Arrays.asList("lambdas","in","action"), (String s) -> s.length());
    

    另外还提供了一些接口

    函数式接口补充

    4. lambda访问局部变量

    访问局部变量

    你可能会问自己,为什么局部变量有这些限制。
    第一,实例变量和局部变量背后的实现有一个关键不同。实例变量都存储在堆中,而局部变量则保存在栈上。如果Lambda可以直接访问局部变量,而且Lambda是在一个线程中使用的,则使用Lambda的线程,可能会在分配该变量的线程将这个变量收回之后,去访问该变量。因此,Java在访问自由局部变量时,实际上是在访问它的副本,而不是访问原始变量。如果局部变量仅仅赋值一次那就没有什么区别了——因此就有了这个限制。
    第二,这一限制不鼓励你使用改变外部变量的典型命令式编程模式(我们会在以后的各章中解释,这种模式会阻碍很容易做到的并行处理)

    闭包

    5. 方法引用

    如何构建方法引用
    方法引用主要有三类。
    (1) 指向静态方法的方法引用(例如 Integer 的 parseInt 方法,写作 Integer::parseInt )。
    (2) 指 向 任 意 类 型 实 例 方 法 的 方 法 引 用 ( 例 如 String 的 length 方 法 , 写 作 String::length )
    (3) 指向现有对象的实例方法的方法引用(假设你有一个局部变量 expensiveTransaction
    用于存放 Transaction 类型的对象,它支持实例方法 getValue ,那么你就可以写 expensive-Transaction::getValue )

    第二种和第三种方法引用可能乍看起来有点儿晕。类似于 String::length 的第二种方法引用的思想就是你在引用一个对象的方法,而这个对象本身是Lambda的一个参数。例如,Lambda表达式 (String s) -> s.toUppeCase() 可以写作 String::toUpperCase 。但第三种方法引用指 的 是 , 你 在 Lambda 中 调 用 一 个 已 经 存 在 的 外 部 对 象 中 的 方 法 。 例 如 , Lambda 表 达 式()->expensiveTransaction.getValue() 可以写作 expensiveTransaction::getValue 。

    方法引用例子
    方法引用实战

    现在我们来尝试实现一个专门的comparator来对一个装满Apple的List进行排序

    1. 传递代码
      因为java已经有了一个sort的接口:
      void sort(Comparator<? super E> c)
      看来我们需要实现一个Comparator的接口才行, 首先我们尝试用最早的匿名类来实现:
    inventory.sort(new Comparator<Apple>() {
      public int compare(Apple a1, Apple a2){
        return a1.getWeight().compareTo(a2.getWeight());
      }
    });
    
    1. 使用lambda表达式
      匿名类的实现太啰嗦了, 我们来尝试用lambda表达式:
    inventory.sort((a1, a2) -> a1.getWeight().compareTo(a2.getWeight()));
    因为Comparator有一个静态方法
    Comparator<Apple> c = Comparator.comparing((Apple a) -> a.getWeight());
    我们就可以再精简为
    inventory.sort(comparing((a) -> a.getWeight()));
    
    1. 使用方法引用
    inventory.sort(comparing(Apple::getWeight));
    

    6. 复合Lambda表达式

    1. 比较器复合:
    inventory.sort(comparing(Apple::getWeight)
    .reversed()
    .thenComparing(Apple::getCountry));
    
    1. 谓词复合:
      谓词接口包括三个方法:negate、and和or,让你可以重用已有的Predicate来创建更复杂的谓词
    Predicate<Apple> redAndHeavyAppleOrGreen = redApple.and(a -> a.getWeight() > 150).or(a -> "green".equals(a.getColor()));
    
    1. 函数复合:
      还可以把Function接口所代表的Lambda表达式复合起来。Function接口为此配了andThen和compose两个默认方法,它们都会返回Function的一个实例。
      Function<Integer, Integer> f = x -> x + 1;
      Function<Integer, Integer> g = x -> x * 2;
      用compose的话,它将意味着f(g(x)),而andThen则意味着g(f(x))

    相关文章

      网友评论

          本文标题:Java8 学习总结 - 基本思想

          本文链接:https://www.haomeiwen.com/subject/mwjgextx.html