java8之如何使用lambda表达式

欢迎访问我的独立博客上的原文http://tantanit.com/java-how-to-use-lambda/ ,转载请务必注明出处。
上一篇文章中介绍了lambda表达式的语法,引入了lambda表达式的使用场景,以及使用lambda 表达式的好处。我们将在这篇文章中,已实例讲解如何定义和使用lambda表达式,以及与其它语言相比,lambda表达式在Java中的特殊规范。

使用匿名内部类的例子

首先明确一点,在Java8出现之前,lambda表达式能够做到的,使用内部类也能做到,lambda表达式只是简化了编程。
下面的例子是从列表中根据条件挑选出读者。

定义TantanitReader:

public class TantanitReader {
    private int age;
    private String loginName;
    private String realName;
    private String career;

    public TantanitReader() {
    }

    public TantanitReader(int age, String loginName, String realName, String career) {
        this.age = age;
        this.loginName = loginName;
        this.realName = realName;
        this.career = career;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getLoginName() {
        return loginName;
    }

    public void setLoginName(String loginName) {
        this.loginName = loginName;
    }

    public String getRealName() {
        return realName;
    }

    public void setRealName(String realName) {
        this.realName = realName;
    }

    public String getCareer() {
        return career;
    }

    public void setCareer(String career) {
        this.career = career;
    }

    @Override
    public String toString() {
        return "age:"+this.getAge()+",loginName:"+this.loginName
        +",realName:"+this.getRealName()+",career:"+this.getCareer();
    }
}

定义判断的接口:

public interface Predicate<T> {
    boolean test(T t);
}

定义选择函数:

public class SelectService<T> {
    public List<T> select(Collection<T> source, Predicate<T> predicate){
        List result = new LinkedList();
        for(T element:source){
            if (predicate.test(element)) {
                result.add(element);
            }
        }
        return result;
    }

}

编写测试用的例子,分别选择成年读者和十多岁(包括10岁)的读者:

public class TantanitReaderPredicateTest {


    public static void main(String[] args) {
        SelectService tantanitReaderSelectSerive
        =new SelectService<TantanitReader>();
        List<TantanitReader> source = new LinkedList<>();
        source.add( new TantanitReader(10,"jack","张三","学生"));
        source.add(new TantanitReader(18,"rose","李四","学生"));
        source.add(new TantanitReader(19,"mike","王五","程序员"));
        source.add(new TantanitReader(20,"jack","赵六","作家"));

        List<TantanitReader> audultReaders
        =tantanitReaderSelectSerive.select(source, new Predicate() {
            @Override
            public boolean test(Object o) {
               TantanitReader tantanitReader=(TantanitReader)o;
               return tantanitReader.getAge()>=18;
            }
        });
        System.out.println("tantanit.com成年读者名单如下:");
        printTantanitReaders(audultReaders);

        System.out.println("tantanit.com 十多岁(包含10岁)成员如下:");
        List<TantanitReader> teenReaders
        =tantanitReaderSelectSerive.select(source, new Predicate() {
            @Override
            public boolean test(Object o) {
                TantanitReader tantanitReader=(TantanitReader)o;
                return tantanitReader.getAge()>=10 && tantanitReader.getAge()<=19;
            }
        });
        printTantanitReaders(teenReaders);
    }


    public static void printTantanitReaders(List<TantanitReader> tantanitReaders) {
        for (TantanitReader tantanitReader : tantanitReaders) {
            System.out.println(tantanitReader.toString());
        }
    }


}

执行后,打印结果如下:

tantanit.com成员读者名单如下:
age:18,loginName:rose,realName:李四,career:学生
age:19,loginName:mike,realName:王五,career:程序员
age:20,loginName:jack,realName:赵六,career:作家
tantanit.com 十多岁(包含10岁)成员如下:
age:10,loginName:jack,realName:张三,career:学生
age:18,loginName:rose,realName:李四,career:学生
age:19,loginName:mike,realName:王五,career:程序员

可以看到,两次选择读者,都需要new Predicate(),并且重写(Override)test方法,而真正的差异其实只在于判断语句:

tantanitReader.getAge()>=18

tantanitReader.getAge()>=10 && tantanitReader.getAge()<=19

但是在Java8之前,由于没有lambda表达式,只能忍受这种冗余。如何用lambda表达式来简化代码呢?

为了照顾Java开发人员既有的编程习惯,与其它语言不同,Java8在设计lambda表达式的使用机制时,规定仍然需要使用接口,并且要求所使用的接口必须是函数式接口,在这个例子中,我们仍然可以使用:

public interface Predicate<T> {
    boolean test(T t);
}

因为这个接口只有一个抽象方法(java8引入了default方法,default方法有具体实现,不算抽象方法),所以它是函数式接口(functional interface)。函数式接口可以加上@FunctionalInterface声明,也可以不加。但是加上之后,编译器在编译阶段就会检查这个接口是否符合函数式接口的定义,所以这里我们定义一个新的接口,并且加上@FunctionalInterface声明:

@FunctionalInterface
public interface PredicateFunction<T> {
    boolean test(T t);
}

并且给SelectService添加一个以PredicateFunction为参数的方法:

public List<T> select(Collection<T> source, PredicateFunction<T> predicate){
    List result = new LinkedList();
    for(T element:source){
        if (predicate.test(element)) {
            result.add(element);
        }
    }
    return result;
}

再修改测试的例子:

public class TantanitReaderPredicateFunctionTest {

    public static void main(String[] args) {
        SelectService tantanitReaderSelectSerive
        =new SelectService<TantanitReader>();
        List<TantanitReader> source = new LinkedList<>();
        source.add( new TantanitReader(10,"jack","张三","学生"));
        source.add(new TantanitReader(18,"rose","李四","学生"));
        source.add(new TantanitReader(19,"mike","王五","程序员"));
        source.add(new TantanitReader(20,"jack","赵六","作家"));

        PredicateFunction<TantanitReader> predicateFunction
         = (TantanitReader tantanitReader) -> tantanitReader.getAge() >= 18;
        List<TantanitReader> audultReaders
        =tantanitReaderSelectSerive.select(source,predicateFunction);

        System.out.println("tantanit.com成员读者名单如下:");
        printTantanitReaders(audultReaders);

        System.out.println("tantanit.com 十多岁(包含10岁)成员如下:");
        PredicateFunction<TantanitReader> predicateFunction2
        = (TantanitReader tantanitReader)
        -> tantanitReader.getAge()>=10 && tantanitReader.getAge()<=19;
        List<TantanitReader> teenReaders
        =tantanitReaderSelectSerive.select(source,predicateFunction2);
        printTantanitReaders(teenReaders);
    }


    public static void printTantanitReaders(List<TantanitReader> tantanitReaders) {
        for (TantanitReader tantanitReader : tantanitReaders) {
            System.out.println(tantanitReader.toString());
        }
    }

}

下面我们分析一下这段代码是如何生效的:

PredicateFunction<TantanitReader> predicateFunction
 = (TantanitReader tantanitReader) -> tantanitReader.getAge() >= 18;
List<TantanitReader> audultReaders
=tantanitReaderSelectSerive.select(source,predicateFunction);

这段代码,生成了一个PredicateFunction类型的实例,并且将该实例的引用作为参数传给tantanitReaderSelectSerive的select方法,并且执行select方法。select在执行过程中,调用predicateFunction的test方法,而test方法的内容就是我们传入的lambda表达式,最终按照lambda表达式,选择出读者。

再进一步,一般可以不定义predicateFunction这个变量,而直接将lambda表达式作为参数传给tantanitReaderSelectSerive的select方法,像这样:

List<TantanitReader> audultReaders
=tantanitReaderSelectSerive.select(
  source,(TantanitReader tantanitReader) -> tantanitReader.getAge() >= 18
);

但是这个例子,实际上会报编译错误,说TantanitReader和tantanitReaderSelectSerive的select方法的定义不匹配,因为select方法使用的是泛型。java8的文档确实是规定了在使用泛型的情况下,不能直接将lambda表达式作为参数,这个挺无语的。如果不使用泛型的,没有这个问题。

小结

下面总结一下如何使用lambda表达式

  1. 首先,定义一个函数式接口(functional interface),并且在接口中定义需要使用的抽象方法。
  2. 编写业务方法,并且以该函数式接口作为参数,并且调用该接口定义的方法,完成业务逻辑。
  3. 调用业务方法,并且将lambda表达式作为参数传入。

如果使用了泛型,最后一步改为先定义一个函数式接口的实例的引用,再作为参数传给业务方法。

此外,lambda表达式还可以继续简化为函数引用,将在后面的文章中讲解。

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

推荐阅读更多精彩内容