java8 lambda in_jdk8-lambda表达式的使用

1, 遍历list集合

List 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接口的一个匿名类

@Testpublic voidtest15() {

List 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() > 4);

}public static void filter(Listnames, Predicate condition) {for(String name : names) {if(condition.test(name)) {

System.out.println(name + " ");

}

}

}

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

为逻辑判断

@Testpublic voidtest16() {

Predicate startWithJ = (n) -> n.startsWith("J");

Predicate fourLength = (n) -> n.length() == 4;

List languages = Arrays.asList("Java", "Scala", "C++", "Haskell", "Lisp");

languages.stream().filter(startWithJ.and(fourLength))

.forEach(System.out::println);

}

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

@Testpublic voidtest14() {

List list = new ArrayList<>();

list.add(1);

list.add(2);

list.add(3);//可改变对象

list.stream().map((i) -> i * 3).forEach(System.out::println);//不可改变元有对象

list.forEach(i -> i = i * 3);

list.forEach(System.out::println);

;

}

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

5.2, flatmap

Stream> inputStream =Stream.of(

Arrays.asList(1),

Arrays.asList(2, 3),

Arrays.asList(4, 5, 6)

);

Stream 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() 方法来做转换

@Testpublic voidtest17() {

List list = new ArrayList<>();

list.add(1);

list.add(2);

list.add(3);

Integer integer= list.stream().map((i) -> i = i * 3)

.reduce((sum, count)-> sum += count).get();

System.out.println(integer);

}

reduce的更多用法

@Testpublic voidtest10() {//字符串连接,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(1, 2, 3, 4).reduce(0, Integer::sum);//求和,sumValue = 10, 无起始值

sumValue = Stream.of(1, 2, 3, 4).reduce(Integer::sum).get();//过滤,字符串连接,concat = "ace"

concat = Stream.of("a", "B", "c", "D", "e", "F").

filter(x-> x.compareTo("Z") > 0).

reduce("", String::concat);

}

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

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

@Test

public void test2() {

List strList = Arrays.asList("abc", "eqwr", "bcd", "qb" , "ehdc", "jk");

List filtered = strList.stream().filter(x -> x.length()> 2).collect(Collectors.toList());

System.out.printf("Original List : %s, filtered list : %s %n", strList, filtered);

}

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

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

@Testpublic voidtest3() {

List 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 numbers = Arrays.asList(9, 10, 3, 4, 7, 3, 4);

List 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 primes = Arrays.asList(2, 3, 5, 7, 11, 13, 17, 19, 23, 29);

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 classPerson {private String name; //姓名

private String location; //地址

publicString getName() {returnname;

}public voidsetName(String name) {this.name =name;

}publicString getLocation() {returnlocation;

}public voidsetLocation(String location) {this.location =location;

}

@OverridepublicString toString() {//TODO Auto-generated method stub

return "Person:" + name + "," +location;

}

}

引用

//使用String默认的排序规则,比较的是Person的name字段

Comparator byName = Comparator.comparing(p ->p.getName());//不用写传入参数,传入的用Person来声明

Comparator byName2 = Comparator.comparing(Person::getName);

12, optional 的两种使用方式

@Testpublic voidtest8() {

String str= "abc";

Optional.ofNullable(str).ifPresent(System.out::println);

}

第二种

@Testpublic voidtest9() {

String str= "abc";//Java 8

Optional.ofNullable(str).map(String::length).orElse(-1);//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$0(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 primes = Arrays.asList(new Integer[]{2, 3,5,7});int factor = 2;

primes.forEach(element-> { System.out.println(factor*element); });

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

lambda眼中的this

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

参考:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值