Java8 实战学习 「方法引用」

Java8 实战学习 方法引用

有时,lambda表达式只会调用现有方法。 在这些情况下,通过名称引用现有方法往往更加清楚。 方法参考使您能够做到这一点; 对于已经有名称的方法,它们是紧凑的,易于阅读的lambda表达式。


方法引用

方法引用让你可以重复使用现有的方法定义,并像Lambda一样传递它们。

书中这么描述方法引用,但是自我感觉方法引用还是蛮难得,至少对我来说。那我们就来看下方法引用到底是什么。

方法引用的一个例子:

(Apple a) -> a.getWeight()

等价于

Apple::getWeight

什么是方法引用

方法引用在书中被称为 Lambda 的语法糖,语法糖大致意思就是一种为了简化代码书写的语法:

语法糖(Syntactic sugar),也译为糖衣语法,是由英国计算机科学家彼得·蘭丁发明的一个术语,指计算机语言中添加的某种语法,这种语法对语言的功能并没有影响,但是更方便程序员使用。

什么意思呢?就是只是为了简化某种代码的写法,而定义的一个新的语法,比如我们 for 循环大多情况下可以使用 foreach 来替换,那么foreach 就是 for 循环的语法糖。而我们用 foreach的时候并没有问太多,因为学习的时候就给我们画上了等号。所以学习方法引用我们可以类比这样的写法。

Lambda 的语法糖就是为了在特定情况下简化 Lambda 书写的语法, wfc! Lambda 看起来都费劲,还要简化,那我写出来的代码是不是就没人能懂了? 个人经过了一天的挣扎后,还是打算拥抱这个变化,这就是一个转变的过程,思想固化后再去接受新的思想需要经历一个过程。

书中对方法引用做了下面的解释:

方法引用可以被看作仅仅 「调用特定方法」 的 Lambda 的一种快捷写法。

它的基本思想是,如果一个Lambda代表的只是“直接调用这个方法”,那最好还是用名称来调用它,而不是去描述如何调用它。

方法引用看作针对「仅仅涉及单一方法的Lambda」的语法糖。

相信大多数人看到这句话以后都有如下的疑问点:

  • 「调用特定方法」的快捷写法,什么方法是特定方法?
  • 「只是“直接调用这个方法”」这个方法?

如果 Lambda 表达式的方法体内,只是调用一个已有的方法,如 ApplegetWeight() 方法,Stringlength() 方法,那么我们就可以把这类的 Lambda 修改成方法引用的方式书写。

这里举个例子方便理解:

假如我们需要将一筐苹果的重量全部统计出来

  1. 我们定义一个方法用来提取指定集合中的苹果的重量:

     private static List<Integer> coverAppleWeight(List<Apple> apples, CoverConsumer consumer) {
            List<Integer> list = new ArrayList<>();
            for (Apple a : apples) {
                list.add(consumer.cover(a));
            }
            return list;
        }
    
  2. 我们需要一个函数接口来完成转换操作,这里的方法签名为 T -> R ,即从一个对象中 选择/提取,我们可以使用 Java 8 提供的 Function 接口或者自己创建一个函数接口:

     public interface CoverConsumer {
        int cover(Apple apple);
     }
    
  3. 使用该方法来提取苹果质量:

       List<Apple> apples = new ArrayList<>();
       apples.add(new Apple(10));
       apples.add(new Apple(11));
       apples.add(new Apple(12));
       apples.add(new Apple(13));
    
       coverAppleWeight(apples, (Apple a) -> a.getWeight());//满足 T->R 的Lambda
    

    满足 T->R 的Lambda 作为 CoverConsumer 的实例传递给 coverAppleWeight 来完成提取操作,具体操作内容就是拿到每个苹果的质量。

    Lambda 的操作在这个时候仅仅

    1. 调用 ApplegetWeight() 方法。 属于「调用特定方法」的范围
    2. 仅仅调用了该方法而没有进行任何其他操作。 属于 「直接调用这个方法」 的范围

    因此我们可以使用方法引用来简化 Lambda :

     coverAppleWeight(apples, Apple::getWeight);//调用 Apple 的指定方法 getWeight 而不做操作
    
  4. 如果我们需要给苹果质量造假,比如我们需要在每个苹果现有的质量加 10 存放如集合,那么还能使用方法引用么,答案是否定的。

    coverAppleWeight(apples, (Apple a) -> a.getWeight() + 10);//不能简化为方法引用调用
    

方法引用的类别

方法引用主要有三类:

  1. 引用「静态方法」 : ContainingClass::staticMethodName
  1. 引用一个特定对象的实例方法:containingObject::instanceMethodName
  1. 引用特定类型的任意对象的实例方法:ContainingType::methodName
  1. 引用构造函数: ClassName::new

第一种方法引用「静态方法的方法引用」:

还记得我们第一天学习的时候,比较两个苹果质量将苹果按质量从大到小排列的例子么?当时我们是这样实现的:

 apples.sort(new ComparatorApple());//List.sort(Comparator<? super E> c)
    
    // 隐藏了无关的代码,直接跳到实现
    public static class ComparatorApple implements Comparator<Apple> {
    
       @Override
       public int compare(Apple a, Apple b) {
           return a.getWeight().compareTo(b.getWeight());
       }
    }
    

现在我们在 Apple 类中添加一个静态方法用来比较两个苹果的重量:

  class Apple {
    
        public String name = "Apple";
        private String color;
    
        Apple(int weight) {
            this.weight = weight;
        }
    
        public int weight;
    
        public Integer getWeight() {
            return weight;
        }
    
        // 用来比较重量的方法
        public static int compareByWeight(Apple a,Apple b){
            return a.getWeight().compareTo(b.getWeight());
        }
    
        public String getColor() {
            return color;
        }
        
    }

有了这个方法 我们之前的 Lambda 表达式可以修改一下:

//  首先修改 ComparatorApple 接口
    
public static class ComparatorApple implements Comparator<Apple> {
    
       @Override
       public int compare(Apple a, Apple b) {
           return Apple.compareByWeight(a,b);//等价于  return a.getWeight().compareTo(b.getWeight());
       }
}
     
// 最终调用的 Lambda 可以就修改如下   
    
apples.sort((Apple a, Apple b) -> Apple.compareByWeight(a, b));
    

我们知道 compareByWeight() 方法是 Apple 的静态方法,实现的功能跟 Comparator 接口要实现的功能相同,所以 Lambda 表达式的签名也是一致的。

所以我们也可以写为:ContainingClass::staticMethodName 格式。

ContainingClass 为包含这个静态方法的类名,而 :: 后则表示方法名 ,值得注意的是这里不需要参数和()。

apples.sort(Apple::compareByWeight);

--

引用一个特定对象的方法

接着筛选苹果的例子说,apples.sort(Comparator<Apple> c) 参数是接收一个满足 Comparator 目标类型的 Lambda 表达式,这个 Lambda 表达式应该满足下面的要求:

(Apple a, Apple b) -> a.getWeight().compareTo(b.getWeight())

这时候我们恰好有一个 比较器提供者ComparisonProvider

  public static class ComparisonProvider {
        public int compareByColor(Apple a, Apple b) {
            return a.getColor().compareTo(b.getColor());
        }

        public static int compareByWeight(Apple a, Apple b) {
            return a.getWeight().compareTo(b.getWeight());
        }
    }

它可以提供各式各样的比较方法,但是并不是一个函数式接口。我们可以将之前的代码修改为:

 ComparisonProvider comparisonProvider = new ComparisonProvider();
 apples.sort((Apple a, Apple b) -> {
            return comparisonProvider.compareByColor(a, b);
        });

在这里 comparisonProvider 就是一个「特定对象」,因为这个对象恰好有比较两个苹果的方法compareByColor,而后者就是这个特定对象的方法。此时我们可以使用方法引用的方式简化 Lambda.

  ComparisonProvider comparisonProvider = new ComparisonProvider();
  apples.sort(comparisonProvider::compareByColor);// 特定对象 :: 对象的方法

这里虽然没有写任何有关于 Apple 的参数,但 JRE 可以推断方法类型参数。 喲~ 不错喔。

--

特殊的类型的任意对象的实例方法引用:ContainingType::methodName

  1. 首先「实例方法」是一个类的实例的方法,而不是静态方法。

  2. 其次特定的类的任意对象 : Lambda表达式的主体中你在引用一个对象的方法,而这个对象恰巧是该 Lambda 的参数的对象

看下边的例子应该好理解:

    // 例子1
  String[] stringArray = {"Barbara", "James", "Mary", "John",
                "Patricia", "Robert", "Michael", "Linda"};
        Arrays.sort(stringArray, String::compareToIgnoreCase);
        List<String> list = Arrays.asList(stringArray);
        list.sort(String::compareToIgnoreCase);
        list.sort((String s, String s1) -> s.compareToIgnoreCase(s1));
  
   // 例子2
    ArrayList<Apple> apples1 = new ArrayList<>();
        apples1.add(new Apple(10));
        apples1.add(new Apple(11));
        apples1.add(new Apple(12));
        apples1.add(new Apple(13));
        apples1.sort((Apple a, Apple b) -> a.compareTo(b));
        apples1.sort(Apple::compareTo);//compareTo是 Apple的一个实例方法
   
   // Apple 中的 compareTo 方法 这里并不是 Compator 的重载方法     
   public int compareTo(Apple apple) {
        return this.getWeight().compareTo(apple.getWeight());
    }


构造参数的方法引用

现在我们可以通过 ClassName::new 来创建一个构造参数的引用。

如果一个构造函数没有参数它适合Supplier的签名() -> Apple。

```
  Supplier<Apple> c1 = Apple::new; Apple a1 = c1.get();
       
 //等价于
  
  Supplier<Apple> c1 = () -> new Apple(); Apple a1 = c1.get();
     
```

如果构造函数的包含一个参数,如是Apple(Integer weight) 它就适合Function接口的签名。

 Function<Integer, Apple> c2 = Apple::new; Apple a2 = c2.apply(110)
        
        //等价于
        
 Function<Integer, Apple> c2 = (weight) -> new Apple(weight); Apple a2 = c2.apply(110);

如果你有一个具有两个参数的构造函数Apple(String color, Integer weight) 那么就可以应用 BiFunction 接口的签名:

BiFunction<String, Integer, Apple> c3 = Apple::new;
Apple c3 = c3.apply("green", 110);

//等价于

BiFunction<String, Integer, Apple> c3 = (color, weight) -> new Apple(color, weight); Apple c3 = c3.apply("green", 110);

如果你需要更多的参数的构造函数,那么你可以自己创建这个函数式接口。

public interface TriFunction<T, U, V, R>{

R apply(T t, U u, V v);

}

TriFunction<Integer, Integer, Integer, Color> colorFactory = Color::new;

关于方法引用的总结:

  1. 方法引用是 Lambda 的语法糖,可以进一步简化 Lambda 的书写。

  2. 方法引用有四种应用场景: 1. 调用静态方法的情况,2.调用一个满足条件的类对象的实例方法(这个对象通常为局部变量) 3. 满足条件的类的任意对象的实例方法 4. 构造参数

  3. 在使用的过程中建议不要直接写出方法引用,通过先写 Lambda 表达式,然后通过编辑器转化为 方法引用的方式。毕竟 IntelliJ(2017.1之后的版本) 这么强大。

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

推荐阅读更多精彩内容