一、Lambda表达式

1.语法格式

Lambda是匿名函数,可以传递代码。使用“->”操作符,改操作符将lambda分成两部分:

左侧:指定了 Lambda 表达式需要的所有参数

右侧:指定了 Lambda 体,即 Lambda 表达式要执行的功能,也就是实现接口方法的代码

  1. // 语法格式一:无参、无返回值  
  2. @Test  
  3. public void test1() {  
  4.     Runnable runable = () -> System.out.println("hello lambda!");  
  5.     runable.run();  
  6. }  
  7.   
  8. // 语法格式二:有一个参、无返回值  
  9. @Test  
  10. public void test2() {  
  11.     Consumer<String> consumer = (args) -> System.out.println("hello!"  
  12.            + args);  
  13.     consumer.accept("lambda");  
  14. }  
  15.   
  16. // 语法格式三:有多个参、有返回值,并且有多条执行语句,用大括号包围  
  17. @Test  
  18. public void test3() {  
  19.    Comparator<Integer> com = (x, y) -> {  
  20.        System.out.println("hello lambda!");  
  21.        return Integer.compare(x, y);  
  22.     };  
  23.     int rs = com.compare(2, 2);  
  24.     System.out.println(rs);  
  25. }  
  26.   
  27. // 语法格式四:右侧如果只有一条执行语句,可以省略大括号和return  
  28. @Test  
  29. public void test4() {  
  30.    Comparator<Integer> com = (Integer x, Integer y) -> Integer.compare(x, y);  
  31.    int rs = com.compare(2, 2);  
  32.    System.out.println(rs);  
  33. }  

注:lambd可以省略接口参数类型,java编译器会根据上下文进行类型推断

2.函数式接口

(1)只包含一个抽象方法的接口,称为函数式接口,可以在任意函数式接口上使用 @FunctionalInterface 注解,lambda需要函数式接口的支持。

(2)java8内置四大核心函数式接口:

Consumer<T>消费型接口:void accept(T t)

  1. @Test  
  2. public void test2() {  
  3.     Consumer<String> consumer = (args) -> System.out.println("hello!"  
  4.            + args);  
  5.     consumer.accept("lambda");  
  6. }  

Supplier<T>供给型接口:T get()

  1. @Test
  2. public void test5() {
  3. List<Integer> rs=getNumList(6,() -> (int)(Math.random()*100));
  4. System.out.println(rs);
  5. }
  6.  
  7. public List<Integer> getNumList(int size,Supplier<Integer> sup){
  8. List<Integer> list=new ArrayList<Integer>();
  9. for (int i = 0; i < size; i++) {
  10. Integer e=sup.get();
  11. list.add(e);
  12. }
  13. return list;
  14. }

Function<T, R>函数型接口:R apply(T t)

  1. @Test  
  2. public void test6() {  
  3. Function<String, Integer> fun=(str)->str.length();  
  4. int len=fun.apply("lambda");  
  5. System.out.println(len);  
  6. }  

Predicate<T>断定型接口:boolean test(T t)

  1. @Test  
  2. public void test7() {  
  3. Predicate<String> check=(str)->str.equals("lambda");  
  4. boolean rs=check.test("lambda");  
  5. System.out.println(rs);  
  6. }  

3.方法引用

当要传递给Lambda体的操作,已经有实现的方法了,可以使用方法引用!( lambda要实现抽象方法的参数列表,必须与方法引用的方法参数列表保持一致! )。
方法引用:使用操作符 “ ::” 将对象或类和方法的名字分隔开来。

三种主要使用情况:

对象::方法名

  1. PrintStream ps=System.out;  
  2. Consumer<String> consumer =  ps::println;  
  3. consumer.accept("lambda");

类::静态方法名

  1. Comparator<Integer> com = Integer::compare;  
  2. int rs = com.compare(2, 2);  
  3. System.out.println(rs);  

类::实例方法名(前提条件:lambda参数列表的第一个参数是实例方法的调用者,第二个参数是实例方法的入参)

  1. BiPredicate<String,String> check=(str1,str2)->str1.equals(str2);  
  2. BiPredicate<String,String> check1=String::equals;  

4.构造器引用

与函数式接口相结合,自动与函数式接口中方法兼容(需要调用的构造器方法参数列表要与函数式接口中方法的参数列表一致)

格式: ClassName::new

5.数组引用

格式: type[] :: new

二、Stream API

1.说明解释

Stream 是 Java8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找、过滤和映射数据等操作。使用Stream API 对集合数据进行操作,就类似于使用 SQL 执行的数据库查询。也可以使用 Stream API 来并行执行操作。简而言之,Stream API 提供了一种高效且易于使用的处理数据的方式。流(Stream)是数据渠道,用于操作数据源(集合、数组等)所生成的元素序列。“集合讲的是数据,流讲的是计算! ”

注意:
①Stream 自己不会存储元素。
②Stream 不会改变源对象。相反,他们会返回一个持有结果的新Stream。
③Stream 操作是延迟执行的。这意味着他们会等到需要结果的时候才执行。

2.Stream 操作步骤

(1)创建Stream

①可以通过Collection系列集合提供的stream()或者parallelStream()获得

  1. List<String> list=new ArrayList<String>();  
  2. Stream<String> stream=list.stream();  

②可以通过Arrays的静态方法stream()获得数组流

  1. Person[] ps=new Person[10];  
  2. Stream<Person> stream=Arrays.stream(ps); 

③可以通过Stream的静态of()

  1. Stream<String> steam=Stream.of("aa","bb","cc"); 

④可以使用静态方法 Stream.iterate() 和Stream.generate(), 创建无限流

(2)中间条件操作

说明:多个中间操作可以连接起来形成一个流水线,除非流水线上触发终止操作,否则中间操作不会执行任何的处理!而在终止操作时一次性全部处理,称为“惰性求值”。

①筛选与切片
        filter——接收 Lambda , 从流中排除某些元素。
        limit——截断流,使其元素不超过给定数量。
        skip(n) —— 跳过元素,返回一个扔掉了前 n 个元素的流。若流中元素不足 n 个,则返回一个空流。与 limit(n) 互补
        distinct——筛选去重,通过流所生成元素的 hashCode() 和 equals() 去除重复元素

  1. List<Person> personList = Arrays.asList(  
  2.                 new Person("李四", 20, 10),  
  3.                 new Person("张三", 40,30),  
  4.                 new Person("王五", 28, 15),  
  5.                 new Person("赵六", 60, 60),  
  6.                 new Person("赵六", 60, 60),  
  7.                 new Person("赵六", 60, 60),  
  8.                 new Person("田七", 8,2)  
  9.         );  
  10.           
  11.         Stream<Person> stream=personList.stream();          
  12.         stream.filter((p)->p.getAge()>20)//过滤保留age>20  
  13.         .limit(5)//只取前两个  
  14.         .skip(1)//跳过前一个,返回剩下的  
  15.         .distinct()//去重,自动定义对象去重要重写equals和hashcode  
  16.         .forEach(System.out::println); 

②映射
        map——接收 Lambda,将元素转换成其他形式或提取信息。接收一个函数作为参数,该函数会被应用到每个元素上,并将其映射成一个新的元素。
        flatMap——接收一个函数作为参数,将流中的每个值都换成另一个流,然后把所有流连接成一个流

  1. Stream<Person> stream=personList.stream();          
  2. stream.map((p)->p.getName())//提取name,组成新的Stream流  
  3.      .forEach(System.out::println);  

③排序
        sorted()——自然排序(Comparable)
        sorted(Comparator com)——定制排序

(3)执行操作

查找与匹配
        allMatch——检查是否匹配所有元素,返回boolean
        anyMatch——检查是否至少匹配一个元素,返回boolean
        noneMatch——检查是否没有匹配的元素,返回boolean
        findFirst——返回第一个元素
        findAny——返回当前流中的任意一个元素
        count——返回流中元素的总个数
        max——返回流中最大值
        min——返回流中最小值

 

  

  

 

java8中Lambda表达式和Stream API的更多相关文章

  1. 十分钟学会Java8:lambda表达式和Stream API

    Java8 的新特性:Lambda表达式.强大的 Stream API.全新时间日期 API.ConcurrentHashMap.MetaSpace.总得来说,Java8 的新特性使 Java 的运行 ...

  2. 十分钟学会Java8的lambda表达式和Stream API

    01:前言一直在用JDK8 ,却从未用过Stream,为了对数组或集合进行一些排序.过滤或数据处理,只会写for循环或者foreach,这就是我曾经的一个写照. 刚开始写写是打基础,但写的多了,各种乏 ...

  3. Java8的lambda表达式和Stream API

    一直在用JDK8 ,却从未用过Stream,为了对数组或集合进行一些排序.过滤或数据处理,只会写for循环或者foreach,这就是我曾经的一个写照. 刚开始写写是打基础,但写的多了,各种乏味,非过来 ...

  4. Java8中Lambda表达式的10个例子

    Java8中Lambda表达式的10个例子 例1 用Lambda表达式实现Runnable接口 //Before Java 8: new Thread(new Runnable() { @Overri ...

  5. java8新特性-lambda表达式和stream API的简单使用

    一.为什么使用lambda Lambda 是一个 匿名函数,我们可以把 Lambda表达式理解为是 一段可以传递的代码(将代码像数据一样进行传递).可以写出更简洁.更灵活的代码.作为一种更紧凑的代码风 ...

  6. JDK1.8新特性(一) ----Lambda表达式、Stream API、函数式接口、方法引用

    jdk1.8新特性知识点: Lambda表达式 Stream API 函数式接口 方法引用和构造器调用 接口中的默认方法和静态方法 新时间日期API default   Lambda表达式     L ...

  7. java8中lambda表达式的应用,以及一些泛型相关

    语法部分就不写了,我们直接抛出一个实际问题,看看java8的这些新特性究竟能给我们带来哪些便利 顺带用到一些泛型编程,一切都是为了简化代码 场景: 一个数据类,用于记录职工信息 public clas ...

  8. 公子奇带你一步一步了解Java8中Lambda表达式

    在上一篇<公子奇带你一步一步了解Java8中行为参数化>中,我们演示到最后将匿名实现简写为 (Police police) -> "浙江".equals(poli ...

  9. JDK1.8中的Lambda表达式和Stream

    1.lambda表达式 Java8最值得学习的特性就是Lambda表达式和Stream API,如果有python或者javascript的语言基础,对理解Lambda表达式有很大帮助,因为Java正 ...

随机推荐

  1. 03 使用Tensorflow做计算题

    我们使用Tensorflow,计算((a+b)*c)^2/a,然后求平方根.看代码: import tensorflow as tf # 输入储存容器 a = tf.placeholder(tf.fl ...

  2. Spring Boot 简介

    作者其他Spring系列文章 Spring Framework简介 Spring框架快速入门  Spring Boot愿景 Spring Boot愿景就是让我们创建可运行的.独立的.基于Spring的 ...

  3. java中Char到底是什么格式的编码

    文本处理中经常有这样的逻辑: String s = new String(bts, "UTF-8"); 看String源代码,里面是一个char[],将bts按照某种编码方式,变成 ...

  4. You must reset your password using ALTER USER statement before executing this statement.

    MySQL 5.7之后,刚初始化的MySQL实例要求先修改密码.否则会报错: mysql> create database test; ERROR 1820 (HY000): You must ...

  5. 应用负载均衡之LVS(三):ipvsadm命令

    */ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...

  6. Go基础系列:Go中的方法

    Go方法简介 Go中的struct结构类似于面向对象中的类.面向对象中,除了成员变量还有方法. Go中也有方法,它是一种特殊的函数,定义于struct之上(与struct关联.绑定),被称为struc ...

  7. spark集群的简单测试和基础命令的使用

    写此篇文章之前,已经搭建好spark集群并测试成功: spark集群搭建文章链接:http://www.cnblogs.com/mmzs/p/8193707.html 一.启动环境 由于每次都要启动, ...

  8. AvosCloud的文件存储 Demo

    时间戳:201310142227 废话少说,直接上代码: package com.dannalapp.main; import com.avos.avoscloud.GetCallback; impo ...

  9. 从零开始学安全(二十二)●PHP日期date参数表

    $date=new DateTime(); echo $date->format("Y-m-d h:i:s");

  10. NGUI 做局部2d卷轴

    网上找到的都是做整个背景的卷轴动画,通常是改变纹理位置或者背景图片的x坐标 没有提到在UI界面里某个部分做卷轴动画,找了很久,才发现NGUI的Panel里的Clipping属性可以裁剪Panel的大小 ...