java lambda表达式学习笔记

时间:2022-09-07 18:46:36

lambda是函数式编程(FP,functional program),在java8中引入,而C#很早之前就有了。在java中lambda表达式是'->',在C#中是‘=>’。

杜甫说:射人先射马,擒贼先擒王。学习一个库要学习它的入口类。lambda的入口类是Stream,一看Stream中的函数就会发现Function,Predicate等lambda元素。

一.几个概念

    函数式接口 Functional Interface,除了static和default类型的方法外,只有一个函数的接口。以前,接口中的一切方法都是public的,现在接口中可以包含default类型的实现方法了。java中没有函数指针的概念,C#中有delegate委托相当于函数指针,但java也是有办法的,用一个类,类里面有一个函数,这个类就相当于函数指针。这么整实现简单,理解简单,但是代码比较冗长。

  谓词 Predicate, 简单来说,谓词就是条件。正规来说,谓词就是一个函数boolean f(x1,x2...),表示变量x1,x2...是否满足条件f。在java中谓词的定义就是一个函数式接口。

函数(映射) Function,将一种类型的对象映射为另一种或同种类型的对象,它就是一个函数ObjectA f(ObjectB)。在java中映射的定义也是一个函数式接口。

@FunctionalInterface
public interface Function<T, R> {
    R apply(T t);
    default <V> Function<V, R> compose(Function<? super V, ? extends T> before) {
        Objects.requireNonNull(before);
        return (V v) -> apply(before.apply(v));
    }
    default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
        Objects.requireNonNull(after);
        return (T t) -> after.apply(apply(t));
    }
    static <T> Function<T, T> identity() {
        return t -> t;
    }
}

可见,除了apply()函数以外,其余default方法在外部都是不可见的。所以定义函数式接口的时候需要把其他函数声明称static或者default类型的。

Optional这个值表示一个对象,这个对象可能为空也可能不为空,对于它可以产生许多行为:

如果它是null,该怎么做orElse()和orElseGet()

如果不为null,该怎么做ifPresent()

判断是否为null,isPresent()

这个类看上去十分鸡肋,但用处十分广泛

package aaa;

import java.util.NoSuchElementException;
import java.util.Optional;

public class OptionalDemo {

      public static void main(String[] args) {
        //创建Optional实例,也可以通过方法返回值得到。
        Optional<String> name = Optional.of("Sanaulla");

        //创建没有值的Optional实例,例如值为'null'
        Optional<Object> empty = Optional.ofNullable(null);

        //isPresent方法用来检查Optional实例是否有值。
        if (name.isPresent()) {
          //调用get()返回Optional值。
          System.out.println(name.get());
        }

        try {
          //在Optional实例上调用get()抛出NoSuchElementException。
          System.out.println(empty.get());
        } catch (NoSuchElementException ex) {
          System.out.println(ex.getMessage());
        }

        //ifPresent方法接受lambda表达式参数。
        //如果Optional值不为空,lambda表达式会处理并在其上执行操作。
        name.ifPresent((value) -> {
          System.out.println("The length of the value is: " + value.length());
        });

        //如果有值orElse方法会返回Optional实例,否则返回传入的错误信息。
        System.out.println(empty.orElse("There is no value present!"));
        System.out.println(name.orElse("There is some value!"));

        //orElseGet与orElse类似,区别在于传入的默认值。
        //orElseGet接受lambda表达式生成默认值。
        System.out.println(empty.orElseGet(() -> "Default Value"));
        System.out.println(name.orElseGet(() -> "Default Value"));

        try {
          //orElseThrow与orElse方法类似,区别在于返回值。
          //orElseThrow抛出由传入的lambda表达式/方法生成异常。
          empty.orElseThrow(Exception::new);
        } catch (Throwable ex) {
          System.out.println(ex.getMessage());
        }

        //map方法通过传入的lambda表达式修改Optonal实例默认值。
        //lambda表达式返回值会包装为Optional实例。
        Optional<String> upperName = name.map((value) -> value.toUpperCase());
        System.out.println(upperName.orElse("No value found"));

        //flatMap与map(Funtion)非常相似,区别在于lambda表达式的返回值。
        //map方法的lambda表达式返回值可以是任何类型,但是返回值会包装成Optional实例。
        //但是flatMap方法的lambda返回值总是Optional类型。
        upperName = name.flatMap((value) -> Optional.of(value.toUpperCase()));
        System.out.println(upperName.orElse("No value found"));

        //filter方法检查Optiona值是否满足给定条件。
        //如果满足返回Optional实例值,否则返回空Optional。
        Optional<String> longName = name.filter((value) -> value.length() > 6);
        System.out.println(longName.orElse("The name is less than 6 characters"));

        //另一个示例,Optional值不满足给定条件。
        Optional<String> anotherName = Optional.of("Sana");
        Optional<String> shortName = anotherName.filter((value) -> value.length() > 6);
        System.out.println(shortName.orElse("The name is less than 6 characters"));

      }

    }

二.lambda表达式

有两个作用

* 作为函数指针

* 替代匿名内部类,替代函数式接口(FunctionalInterface)

lambda不是语法糖,它在内部实现上也跟匿名内部类不同,匿名内部类需要进行类文件加载,而lambda表达式不用,所以lambda表达式效率比匿名内部类高。

三种用法

用'(x1,x2,x3)'表示传入参数

如果无参写作’()‘

参数类型可以指明,也可以不指明,java会根据后半部分函数形参自动推断出来。

        List<String> a = Arrays.asList("we i di ao is great".split(" "));
        a.forEach((s) -> System.out.println(s));// 表达式
        a.forEach((String s) -> {
            System.out.println(s);
        });// 语句块
        a.forEach(System.out::println);// 函数

方法引用

object::fun()

className::fun()静态方法引用

className::new  构造函数引用

public class LambdaIntro {
    // functional interface 函数式接口
    public static interface ItemWithIndexVisitor<E> {
        public void visit(E item, int index);
    }

    public static <E> void eachWithIndex(List<E> list,
            ItemWithIndexVisitor<E> visitor) {
        for (int i = 0; i < list.size(); i++) {
            visitor.visit(list.get(i), i);
        }
    }
    // 一个普通函数,用作函数指针
    public static <E> void printItem(E value, int index) {
        String output = String.format("%d -> %s", index, value.toString());
        System.out.println(output);
    }
    public static void main(String[] args) {
        List<String> list = Arrays.asList("A", "B", "C");
        // 第一种方式
        eachWithIndex(list, (value, index) -> {
            String output = String.format("%d -> %s", index, value);
            System.out.println(output);
        });
        // 第二种方式
        eachWithIndex(list, LambdaIntro::printItem);
    }
}

三.使用Stream

创建Stream的两种方式

* Stream接口的工厂方法

* 集合框架的stream()函数

首先来了解使用Stream接口来创建Stream,可以创建三种流:普通枚举流,产生器,迭代器。

        //of:通过枚举方式创建流
        Stream<Integer> one = Stream.of(1, 2, 3);
        //流是可以拼接的,从而产生新流
        Stream<Integer> two = Stream.concat(one, Stream.of(4, 5, 6));
        two.forEach(System.out::println);
        //逐个加入,那就用Builder构建器来实现
        Builder<Integer> builder = Stream.builder();
        Stream<Integer> three = builder.add(3).add(4).build();
        three.forEach((s) -> System.out.println(s));
        //产生器流generator
        Random random = new Random();
        Stream<Integer> four = Stream.generate(() -> random.nextInt());
        four.limit(10).forEach(System.out::println);
        //迭代器iterator,UnaryOperator一元运算符可以通过lambda表达式来创建
        Stream<Integer>five=Stream.iterate(2, new UnaryOperator<Integer>() {
            @Override
            public Integer apply(Integer t) {
                return t = (t * 5 + 7) % 13;
            }
        });
        five.limit(10).forEach(System.out::println);

注意产生器generator和迭代器iterator是无限输出的,可以用limit来约束之。

集合框架都继承了Collection接口,而Collection接口就有一个stream()函数。所以剩下的任务就是如何利用流的强大特性来写出优雅的代码来。

要想深刻的了解Stream的一些函数,那就先不要使用lambda表达式,一旦了解它的普通实现,很容易改写成lambda的形式。

流Stream中的函数明显分为两类,一类返回值还是Stream,可以继续用流来处理,另一类返回值不是Stream,不能再用流中函数处理了。

Stream filter(Predicate)删除掉流中不满足条件的元素并返回新的Stream

Stream map(Function)映射,把流中的元素映射一下变成一个新流,还有mapToInt(),mapToLong(),mapToDouble()等函数,它们终究还是映射,只是映射结果更单一。map是一对一映射,flatMap是一对多映射。把一个元素映射成多个元素

        Arrays.asList(1, 2, 3).stream()
                .flatMap(new Function<Integer, Stream<Integer>>() {
                    @Override
                    public Stream<Integer> apply(Integer t) {
                        return Arrays.asList(t, t + 10, t + 100).stream();
                    }
                }).forEach(System.out::println);

输出为1 11 101 2 12 102 3 13 103   中间我省略了换行符

distinct()去除流中重复元素

sorted()和sorted(Comparator cmp)对流中元素排序

peek()弹出一个元素

        Stream.of("one", "two", "three", "four").filter(e -> e.length() > 3)
                .peek(e -> System.out.println("Filtered value: " + e))
                .map(String::toUpperCase)
                .peek(e -> System.out.println("Mapped value: " + e))
                .collect(Collectors.toList());

执行结果

Filtered value: three
Mapped value: THREE
Filtered value: four
Mapped value: FOUR

limit(int cnt)只返回cnt个元素,skip(int cnt)跳过cnt个元素。

forEach(Consumer consumer)对于每一个元素都执行某种操作

reduce()将多个值映射为一个值,实现多对一映射

         Stream.of("one", "two", "three", "four")
                .reduce(new BinaryOperator<String>() {
                    @Override
                    public String apply(String t, String u) {
                        System.out.println(t+":"+u);
                        return t + "," + u;
                    }
                }).ifPresent(System.out::println);

输出为

one:two
one,two:three
one,two,three:four
one,two,three,four

可见,apply(t,u)函数中的t表示当前总量,u表示当前元素。

reduce(T identity,BinaryOperator<T>f)表示带初始值的reduce,比如求和函数,如果identity=9,表示一开始sum=9,此函数返回具体的对象。

        String s = Stream.of("one", "two", "three", "four").reduce("baga",
                new BinaryOperator<String>() {

                    @Override
                    public String apply(String t, String u) {
                        return t +","+ u;
                    }
                });
        System.out.println(s);

输出:

baga,one,two,three,four

collect()

Map<String, Map<String, List<Person>>> peopleByStateAndCity
         = personStream.collect(Collectors.groupingBy(Person::getState,
                                                      Collectors.groupingBy(Person::getCity)));
        String s=Stream.of("one", "two", "three", "four")
                .collect(Collectors.joining(","));
        System.out.println(s);

输出one,two,three,four

Collectors包含许多有用的静态方法

聚集函数min(),max(),count()很像sql中的聚集函数

匹配函数allMatch(Predicate p),anyMatch(Predicate p),noneMath(Predicate p)流中全部匹配,部分匹配,完全不匹配,返回布尔值

java lambda表达式学习笔记的更多相关文章

  1. C&num; Lambda表达式学习笔记

    本笔记摘抄自:https://www.cnblogs.com/leslies2/archive/2012/03/22/2389318.html,记录一下学习过程以备后续查用.     一.Lambda ...

  2. lambda 表达式学习笔记

    在Java中传递一个代码段并不容易,不能直接传递代码段.Java是一个面向对象语言,所以必须构造一个对象,这个对象的类需要一个方法能包含所需的代码.lambda的出现有效的解决这个问题,让代码变得更加 ...

  3. python函数和lambda表达式学习笔记

    1. python函数 不同于其他语言,python支持函数返回多个值 为函数提供说明文档:help(函数名)或者函数名.__doc__ def str_max(str1, str2): ''' 比较 ...

  4. Lambda表达式学习笔记

    Lambda基础语法 Java8中引入了一个新的操作符" -> ",该操作符被称为箭头操作符或Lambda操作符,箭头操作符将Lambda表达式拆分成两部分: 左侧:Lamb ...

  5. java 8 中lambda表达式学习

    转自 http://blog.csdn.net/renfufei/article/details/24600507 http://www.jdon.com/idea/java/10-example-o ...

  6. Java Lambda表达式初探

    Java Lambda表达式初探 前言 本文受启发于Trisha Gee在JavaOne 2016的主题演讲Refactoring to Java 8. Java 8已经发行两年多,但很多人仍然在使用 ...

  7. Java Lambda表达式入门&lbrack;转&rsqb;

    原文链接: Start Using Java Lambda Expressions http://blog.csdn.net/renfufei/article/details/24600507 下载示 ...

  8. 【Java】「深入理解Java虚拟机」学习笔记(1) - Java语言发展趋势

    0.前言 从这篇随笔开始记录Java虚拟机的内容,以前只是对Java的应用,聚焦的是业务,了解的只是语言层面,现在想深入学习一下. 对JVM的学习肯定不是看一遍书就能掌握的,在今后的学习和实践中如果有 ...

  9. 《深入理解Java虚拟机》学习笔记

    <深入理解Java虚拟机>学习笔记 一.走近Java JDK(Java Development Kit):包含Java程序设计语言,Java虚拟机,JavaAPI,是用于支持 Java 程 ...

随机推荐

  1. 用EmEditor实现PDF转Word后的对齐排版

    Redraw = false//禁止重绘(类似于VBA中的: Application.screenupdating=FALSE),以提高运行效率 //去除所有空行和只由空白字符构成的行 documen ...

  2. ubuntu 16&period;04安装docker

    环境 操作系统:ubuntu 16.04 64位,默认安装 准备 1. 添加GPG key: $ sudo apt-key adv --keyserver hkp://p80.pool.sks-key ...

  3. &period;NET足球赛事资料数据库平台SmartLottery开源发布——全球足球联赛应有尽有

            本博客所有文章分类的总目录:[总目录]本博客博文总目录-实时更新 开源C#彩票数据资料库系列文章总目录:[目录]C#搭建足球赛事资料库与预测平台与彩票数据分析目录 前2个月,我的系列文 ...

  4. trie树的建立方法汇总

    方法一:孩子兄弟表示法 即对于某一个点,记录他的第一个孩子以及他的同父亲的下一个儿子. 具体代码如下: #include <cstdio> #include <cstring> ...

  5. &lbrack;Visual Studio Online&rsqb; 移除Work Item&lpar;Feature、Backlog item、Task&rpar;

    [Visual Studio Online] 移除Work Item(Feature.Backlog item.Task) 移除 项目的开发过程中,使用Visual Studio Online来做Sc ...

  6. MVC的URL路由规则

    MVC的URL路由规则 Routing的作用:它首先是获取到View传过来的请求,并解析Url请求中Controller和Action以及数据,其次他将识别出来的数据传递给Controller的Act ...

  7. javascript操作写入txt文件及消息&colon; Automation 服务器不能创建对象问题

    简单的写入txt代码: function WriteTxt() {      var fso, tf;      fso = new ActiveXObject("Scripting.Fil ...

  8. yii2 对象跟数组输出数据到view视图方法

    public function actionJiekou(){ $url = 'http://wap.guoshihui.com/_static/wap/video/startVideo.mp4'; ...

  9. 树莓派&period;Qt&period;打包开发好的程序并运行的方法

    Qt开发的软件, 想要部署在树莓派上运行, 需要进行打包和发布 主要步骤如下: 1. 找1个树莓派用于开发与打包, 所以需要在它上面安装Qt开发环境 树莓派上安装Qt的方法, 可以看这里>&gt ...

  10. 愤怒的小鸟【&dollar;DP&dollar;优化】

    卡常的状压\(DP\),愤怒的小鸟. 其实本来是个很水的状压\(DP\),但因为最后三个点\(n=18\),成功地把我的不可能达到的下界为\(\Omega(2^nn^2)\),紧确的上界为\(O(2^ ...