1, 遍历list集合

   List<Integer> list = new ArrayList<>();
  list.add(1);
  list.add(2);
  list.add(3);
  // 直接打印
  list.forEach(System.out::println);   // 取值分别操作
  list.forEach(i -> {
    System.out.println(i * 3);
  });

2, 利用函数式接口实现匿名内部类

实现一个多线程

new Thread( () -> System.out.println("In Java8!") ).start();

还可以

(params) -> expression
(params) -> statement
(params) -> { statements }

3, Predicate接口

Predicate是jdk8 中的新增接口, 共有5个方法,

//Returns a predicate which evaluates to true only if this predicate
//and the provided predicate both evaluate to true.
and(Predicate<? super T> p) //Returns a predicate which negates the result of this predicate.
negate() //Returns a predicate which evaluates to true if either
//this predicate or the provided predicate evaluates to true
or(Predicate<? super T> p) //Returns true if the input object matches some criteria
test(T t) //Returns a predicate that evaluates to true if both or neither
//of the component predicates evaluate to true
xor(Predicate<? super T> p)

该接口除了test方法是抽象方法, 其余都是default方法, 该接口可接受一个 lambda表达式, 其实就是实现了test接口的一个匿名类

    @Test
public void test15() {
List<String> languages = Arrays.asList("Java", "Scala", "C++", "Haskell", "Lisp"); System.out.println("Languages which starts with J :");
filter(languages, (str) -> ((String) str).startsWith("J")); System.out.println("Languages which ends with a ");
filter(languages, (str) -> ((String) str).endsWith("a")); System.out.println("Print all languages :");
filter(languages, (str) -> true); System.out.println("Print no language : ");
filter(languages, (str) -> false); System.out.println("Print language whose length greater than 4:");
filter(languages, (str) -> ((String) str).length() > ); } public static void filter(List<String> names, Predicate condition) {
for (String name : names) {
if (condition.test(name)) {
System.out.println(name + " ");
}
}
}

4,Predicate接口中的 and, or, xor的使用

为逻辑判断

    @Test
public void test16() {
Predicate<String> startWithJ = (n) -> n.startsWith("J");
Predicate<String> fourLength = (n) -> n.length() == ; List<String> languages = Arrays.asList("Java", "Scala", "C++", "Haskell", "Lisp");
languages.stream().filter(startWithJ.and(fourLength))
.forEach(System.out::println);
}

5, map, 允许将对象进行转换, 比如, 可以更改list中的每个元素的值

    @Test
public void test14() {
List<Integer> list = new ArrayList<>();
list.add();
list.add();
list.add(); // 可改变对象
list.stream().map((i) -> i * ).forEach(System.out::println); // 不可改变元有对象
list.forEach(i -> i = i * );
list.forEach(System.out::println);
;
}

只在本次调用中有效, 并不会改变原有的list

5.2, flatmap

Stream<List<Integer>> inputStream = Stream.of(
Arrays.asList(),
Arrays.asList(, ),
Arrays.asList(, , )
);
Stream<Integer> outputStream = inputStream.
flatMap((childList) -> childList.stream());

6, reduce, 用来将值进行合并, 又称折叠操作, Map和Reduce操作是函数式编程的核心操作

SQL中类似 sum()、avg() 或者 count() 的聚集函数,实际上就是 reduce 操作,因为它们接收多个值并返回一个值。流API定义的 reduceh() 函数可以接受lambda表达式,并对所有值进行合并。IntStream这样的类有类似 average()、count()、sum() 的内建方法来做 reduce 操作,也有mapToLong()、mapToDouble() 方法来做转换

    @Test
public void test17() {
List<Integer> list = new ArrayList<>();
list.add();
list.add();
list.add(); Integer integer = list.stream().map((i) -> i = i * )
.reduce((sum, count) -> sum += count).get(); System.out.println(integer);
}

reduce的更多用法

    @Test
public void test10() {
// 字符串连接,concat = "ABCD"
String concat = Stream.of("A", "B", "C", "D").reduce("", String::concat);
// 求最小值,minValue = -3.0
double minValue = Stream.of(-1.5, 1.0, -3.0, -2.0).reduce(Double.MAX_VALUE, Double::min);
// 求和,sumValue = 10, 有起始值
int sumValue = Stream.of(, , , ).reduce(, Integer::sum);
// 求和,sumValue = 10, 无起始值
sumValue = Stream.of(, , , ).reduce(Integer::sum).get();
// 过滤,字符串连接,concat = "ace"
concat = Stream.of("a", "B", "c", "D", "e", "F").
filter(x -> x.compareTo("Z") > ).
reduce("", String::concat);
}

7, 通过过滤创建一个string, list

过滤是Java开发者在大规模集合上的一个常用操作,而现在使用lambda表达式和流API过滤大规模数据集合是惊人的简单。流提供了一个 filter() 方法,接受一个 Predicate 对象,即可以传入一个lambda表达式作为过滤逻辑。下面的例子是用lambda表达式过滤Java集合,将帮助理解。

@Test
public void test2() {
  List<String> strList = Arrays.asList("abc", "eqwr", "bcd", "qb" , "ehdc", "jk");
  List<String> filtered = strList.stream().filter(x -> x.length()> 2).collect(Collectors.toList());
  System.out.printf("Original List : %s, filtered list : %s %n", strList, filtered);
}

过滤后将获取一个新的列表

8, 对列表的每个元素使用 函数

    @Test
public void test3() {
List<String> strList = Arrays.asList("abc", "eqwr", "bcd", "qb" , "ehdc", "jk");
String collect = strList.stream().map(x -> x.toUpperCase()).collect(Collectors.joining(", "));
System.out.printf("filtered list : %s %n", collect);
}

9, 使用distinct进行去重

List<Integer> numbers = Arrays.asList(, , , , , , );
List<Integer> distinct = numbers.stream().map( i -> i*i).distinct().collect(Collectors.toList());
System.out.printf("Original List : %s, Square Without duplicates : %s %n", numbers, distinct);

10, 计算最值和平均值

IntStream、LongStream 和 DoubleStream 等流的类中,有个非常有用的方法叫做 summaryStatistics() 。可以返回 IntSummaryStatistics、LongSummaryStatistics 或者 DoubleSummaryStatistic s,描述流中元素的各种摘要数据。在本例中,我们用这个方法来计算列表的最大值和最小值。它也有 getSum() 和 getAverage() 方法来获得列表的所有元素的总和及平均值。

//获取数字的个数、最小值、最大值、总和以及平均值
List<Integer> primes = Arrays.asList(, , , , , , , , , );
IntSummaryStatistics stats = primes.stream().mapToInt((x) -> x).summaryStatistics();
System.out.println("Highest prime number in List : " + stats.getMax());
System.out.println("Lowest prime number in List : " + stats.getMin());
System.out.println("Sum of all prime numbers : " + stats.getSum());
System.out.println("Average of all prime numbers : " + stats.getAverage());

11, 使用方法引用, 不对参数做任何修改

方法引用有3种形式

//把lambda表达式的参数直接当成instanceMethod|staticMethod的参数来调用。比如System.out::println等同于x->System.out.println(x);Math::max等同于(x, y)->Math.max(x,y)。
objectName::instanceMethod
ClassName::staticMethod // 把lambda表达式的第一个参数当成instanceMethod的目标对象,其他剩余参数当成该方法的参数。比如String::toLowerCase等同于x->x.toLowerCase()。
ClassName::instanceMethod

Person

package com.lambda.usebean;

/**
* 实体类Person
* @author MingChenchen
*
*/
public class Person {
private String name; //姓名
private String location; //地址 public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getLocation() {
return location;
}
public void setLocation(String location) {
this.location = location;
} @Override
public String toString() {
// TODO Auto-generated method stub
return "Person:" + name + "," + location;
}
}

引用

//使用String默认的排序规则,比较的是Person的name字段
Comparator<Person> byName = Comparator.comparing(p -> p.getName());
//不用写传入参数,传入的用Person来声明
Comparator<Person> byName2 = Comparator.comparing(Person::getName);

12, optional 的两种使用方式

    @Test
public void test8() {
String str = "abc";
Optional.ofNullable(str).ifPresent(System.out::println);
}

第二种

    @Test
public void test9() {
String str = "abc";
// Java 8
Optional.ofNullable(str).map(String::length).orElse(-);
// Pre-Java 8
// return if (text != null) ? text.length() : -1;
}

关于lambda的注意:

1)lambda表达式仅能放入如下代码:预定义使用了 @Functional 注释的函数式接口,自带一个抽象函数的方法,或者SAM(Single Abstract Method 单个抽象方法)类型。这些称为lambda表达式的目标类型,可以用作返回类型,或lambda目标代码的参数。例如,若一个方法接收Runnable、Comparable或者 Callable 接口,都有单个抽象方法,可以传入lambda表达式。类似的,如果一个方法接受声明于 java.util.function 包内的接口,例如 Predicate、Function、Consumer 或 Supplier,那么可以向其传lambda表达式。

2)lambda表达式内可以使用方法引用,仅当该方法不修改lambda表达式提供的参数。本例中的lambda表达式可以换为方法引用,因为这仅是一个参数相同的简单方法调用。

list.forEach(n -> System.out.println(n));
list.forEach(System.out::println); // 使用方法引用

然而,若对参数有任何修改,则不能使用方法引用,而需键入完整地lambda表达式,如下所示:

list.forEach((String s) -> System.out.println("*" + s + "*"));

事实上,可以省略这里的lambda参数的类型声明,编译器可以从列表的类属性推测出来。

3)lambda内部可以使用静态、非静态和局部变量,这称为lambda内的变量捕获。

4)Lambda表达式在Java中又称为闭包或匿名函数,所以如果有同事把它叫闭包的时候,不用惊讶。

5)Lambda方法在编译器内部被翻译成私有方法,并派发 invokedynamic 字节码指令来进行调用。可以使用JDK中的 javap 工具来反编译class文件。使用 javap -p 或 javap -c -v 命令来看一看lambda表达式生成的字节码。大致应该长这样:

private static java.lang.Object lambda$(java.lang.String);

6)lambda表达式有个限制,那就是只能引用 final 或 final 局部变量,这就是说不能在lambda内部修改定义在域外的变量。

Compile time error : "local variables referenced from a lambda expression must be final or effectively final"

另外,只是访问它而不作修改是可以的,如下所示:

List<Integer> primes = Arrays.asList(new Integer[]{, ,,});
int factor = ;
primes.forEach(element -> { System.out.println(factor*element); });

因此,它看起来更像不可变闭包,类似于Python。

lambda眼中的this

在lambda中,this不是指向lambda表达式产生的那个SAM对象,而是声明它的外部对象。

参考:

http://www.importnew.com/16436.html

http://ifeve.com/stream/

http://ifeve.com/lambda/

http://blog.csdn.net/jinzhencs/article/details/50748202

http://ifeve.com/predicate-and-consumer-interface-in-java-util-function-package-in-java-8/

http://www.jdon.com/idea/java/10-example-of-lambda-expressions-in-java8.html

jdk8-lambda表达式的使用的更多相关文章

  1. jdk8 Lambda表达式与匿名内部类比较

    Labmda表达式与匿名内部类 前言 Java Labmda表达式的一个重要用法是简化某些匿名内部类(Anonymous Classes)的写法.实际上Lambda表达式并不仅仅是匿名内部类的语法糖, ...

  2. JDK8 Lambda表达式对代码的简化

    只是举个例子: public class LambdaDemo { public static String findData( String name , LambdaInterface finde ...

  3. jdk8 lambda表达式总结

    Java8 lambda表达式10个示例   1. 实现Runnable线程案例 使用() -> {} 替代匿名类: //Before Java 8: new Thread(new Runnab ...

  4. Lambda表达式和匿名内部类(I)

    本文git地址 前言 Java Labmda表达式的一个重要用法是简化某些匿名内部类(Anonymous Classes)的写法.实际上Lambda表达式并不仅仅是匿名内部类的语法糖,JVM内部是通过 ...

  5. Java基础学习总结(69)——匿名内部类与Lambda表达式

    前言 Java Labmda表达式的一个重要用法是简化某些匿名内部类(Anonymous Classes)的写法.实际上Lambda表达式并不仅仅是匿名内部类的语法糖,JVM内部是通过invokedy ...

  6. JDK8 的 Lambda 表达式原理

    JDK8 使用一行 Lambda 表达式可以代替先前用匿名类五六行代码所做的事情,那么它是怎么实现的呢?从所周知,匿名类会在编译的时候生成与宿主类带上 $1, $2 的类文件,如写在 TestLamb ...

  7. jdk8的新特性 Lambda表达式

    很多同学一开始接触Java8可能对Java8 Lambda表达式有点陌生. //这是一个普通的集合 List<Employee> list = em.selectEmployeeByLog ...

  8. JDK8的新特性——Lambda表达式

    JDK8已经发布快4年的时间了,现在来谈它的新特性显得略微的有点“不合时宜”.尽管JDK8已不再“新”,但它的重要特性之一——Lambda表达式依然是不被大部分开发者所熟练运用,甚至不被开发者所熟知. ...

  9. jdk8新特性-亮瞎眼的lambda表达式

    jdk8之前,尤其是在写GUI程序的事件监听的时候,各种的匿名内部类,大把大把拖沓的代码,程序毫无美感可言!既然Java中一切皆为对象,那么,就类似于某些动态语言一样,函数也可以当成是对象啊!代码块也 ...

  10. JDK8 新特性 Lambda表达式

    1.java8中Lambda表达式基础语法: (x,y) -> {} 左侧是一个小括号,里面是要实现的抽象方法的参数,有几个参数就写几个参数名,无参可写空括号,无需声明参数类型: 中间是一个jd ...

随机推荐

  1. (求凹包) Bicycle Race (CF 659D) 简单题

    http://codeforces.com/contest/659/problem/D     Maria participates in a bicycle race. The speedway t ...

  2. 我们学习Java今后的发展前景

      Java应用的前景   Java技术自1995年问世以来,在我国的应用和开发也得到了迅速普及,总体来看,这 些应用主要集中于企业应用开发.据有关单位调查显示,从开发领域的分布情况上看Web 开发占 ...

  3. CAD:计算三角形的外接圆圆心

    条件:三个定点不共线

  4. Android-Genymotion Unable to load VirtualBox engine

    点击Genymotion出现Genymotion Unable to load VirtualBox engine 解决方案: 打开官网,下载VirtualBox 点击安装: 然后再次点击打开Geny ...

  5. java基础梳理

  6. 10 个免费的Bootstrap Admin 主题,模板收集

    In designing websites today, one of the must have frameworks is the twitter bootstrap. To those who ...

  7. 数据库操作类《SqlHelper》

    一.背景 在看了一本书叫<Visual Studio 2010(C#)Windows数据库项目开发>后,觉得很多编程技术需要积累,因为一个简单的项目里包含的技术太多了,容易忘记.每次需要用 ...

  8. Git入门--创建版本库,关联远程库,从远程库下载

    1.(先进入项目文件夹)通过命令 git init 把这个目录变成git可以管理的仓库 git init 2.把文件添加到版本库中,使用命令 git add .添加到暂存区里面去,不要忘记后面的小数点 ...

  9. 【文文殿下】 [SDOI2013]保护出题人 题解

    题解 我们把伤害-时间图像画出来.然后维护一下僵尸血量的前缀和.最好情况肯定是有一个僵尸恰好死在戴夫家门口.我们把原点到其他n个点的斜率最大的一个累积到答案. 发现每添加一个点,其他所有点的坐标都变了 ...

  10. lua 源码阅读顺序

    https://www.reddit.com/comments/63hth/ask_reddit_which_oss_codebases_out_there_are_so/c02pxbp Online ...