浅谈lambda表达式<最通俗易懂的讲解
Java8发布已经有一段时间了,这次发布的改动比较大,很多人将这次改动与Java5的升级相提并论。Java8其中一个很重要的新特性就是lambda表达式,允许我们将行为传到函数中。想想看,在Java8 之前我们想要将行为传入函数,仅有的选择就是匿名内部类。Java8发布以后,lambda表达式将大量替代匿名内部类的使用,简化代码的同时,更突出了原来匿名内部类中最重要的那部分包含真正逻辑的代码。尤其是对于做数据的同学来说,当习惯使用类似scala之类的函数式编程语言以后,体会将更加深刻。现在我们就来看看Java8中lambda表达式的一些常见写法。
lambda体中调用方法的参数列表和返回值类型,要和函数式接口中抽象方法的参数列表和返回值类型保持一致。
一、替代匿名内部类
lambda表达式用的最多的场合就是替代匿名内部类,实现Runnable接口是匿名内部类的经典例子。lambda表达式的功能相当强大,用()->就可以代替整个匿名内部类!
package OSChina.Lambda;
import org.junit.Test;
import java.util.Comparator;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
public class Test1{
public static void main(String[] args) {
Runnable runnable = new Runnable() {
@Override
public void run() {
System.out.println("普通,线程启动");
}
};
runnable.run();
test2();
test3();
test4();
test5();
}
//无参数,无返回值
public static void test2() {
//“->”左边只有一个小括号,表示无参数,右边是Lambda体(就相当于实现了匿名内部类里面的方法了,(即就是一个可用的接口实现类了。))
Runnable runnable = ()->System.out.println("Lambda 表达式方式,线程启动");
runnable.run();
}
//有一个参数,并且无返回值
public static void test3() {
//这个e就代表所实现的接口的方法的参数,
Consumer<String> consumer = e->System.out.println("Lambda 表达式方式,"+e);
consumer.accept("传入参数");
}
//有两个以上的参数,有返回值,并且 Lambda 体中有多条语句
public static void test4() {
//Lambda 体中有多条语句,记得要用大括号括起来
Comparator<Integer> com = (x, y) -> {
System.out.println("函数式接口");
return Integer.compare(x, y);
};
int compare = com.compare(100, 244);
System.out.println("有两个以上的参数,有返回值,"+compare);
}
//若 Lambda 体中只有一条语句, return 和 大括号都可以省略不写
public static void test5() {
//Comparator com = (x, y) -> Integer.compare(100, 244);
System.out.println("若 Lambda 体中只有一条语句, return 和 大括号都可以省略不写,"+Integer.compare(100, 244));
}
}
二、Java8四大内置函数式接口
如果使用lambda还要自己写一个接口的话太麻烦,所以Java自己提供了一些接口:
1、Consumer 消费性接口:void accept(T t);
//有一个参数,并且无返回值
public static void test3() {
//这个e就代表所实现的接口的方法的参数,
Consumer<String> consumer = e->System.out.println("Lambda 表达式方式,"+e);
consumer.accept("传入参数");
}
2、Supplier供给型接口:T get();
package OSChina.Lambda;
import java.util.ArrayList;
import java.util.function.Supplier;
public class Test2 {
public static void main(String[] args) {
ArrayList<Integer> res = getNumList(10,()->(int)(Math.random()*100));
System.out.println(res);
}
public static ArrayList<Integer> getNumList(int num, Supplier<Integer> sup){
ArrayList<Integer> list = new ArrayList<>();
for (int i = 0; i < num; i++) {
Integer e = sup.get();
list.add(e);
}
return list;
}
}
3、Function 函数式接口:R apply(T t);
package OSChina.Lambda;
import java.util.function.Function;
public class Test2 {
public static void main(String[] args) {
String newStr = strHandler("abc",(str)->str.toUpperCase());
System.out.println(newStr);
newStr = strHandler(" abc ",(str)->str.trim());
System.out.println(newStr);
}
public static String strHandler(String str, Function<String,String>fun){
return fun.apply(str);
}
}
4、Predicate 断言式接口:boolean test(T t);
判断一些字符串数组判断长度>2的字符串:
package OSChina.Lambda;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
public class Test2 {
public static void main(String[] args) {
List<String> list = Arrays.asList("hello","jiangshuying","lambda","www","ok","q");
List<String> ret = filterStr(list,(str)->str.length()>2);
System.out.println(ret);
}
public static List<String> filterStr(List<String> list, Predicate<String> pre){
ArrayList<String> arrayList = new ArrayList<>();
for(String str:list){
if(pre.test(str)) {
arrayList.add(str);
}
}
return arrayList;
}
}
三、方法引用与构造器引用
要求:实现抽象方法的参数列表和返回值类型,必须与方法引用的方法的参数列表和返回值类型保持一致!
方法引用:使用操作符“::”将类与方法分隔开来。
对象::实例方法名
类::静态方法名
类::实例方法名
举个例子:
public static void test9(){
Comparator<Integer> comparator = (x,y)->Integer.compare(x,y);
Comparator<Integer> comparator1 = Integer::compare;
int compare = comparator.compare(1,2);
int compare1 = comparator1.compare(1,2);
System.out.println("compare:"+compare);
System.out.println("compare1:"+compare1);
}
四、lambda表达式的一些常见用法
1、使用lambda表达式对集合进行迭代
package OSChina.Lambda;
import java.util.Arrays;
import java.util.List;
public class Test3 {
public static void main(String[] args) {
List<String> list = Arrays.asList("java","c#","javascript");
//before java8
for (String str:list){
System.out.println("before java8,"+str);
}
//after java8
list.forEach(x-> System.out.println("after java8,"+x));
}
}
2、用lambda表达式实现map
map函数可以说是函数式编程里最重要的一个方法了。map的作用是将一个对象变换为另外一个。在我们的例子中,就是通过map方法将cost增加了0,05倍的大小然后输出。
package OSChina.Lambda;
import java.util.Arrays;
import java.util.List;
public class Test3 {
public static void main(String[] args) {
List<Double> list = Arrays.asList(10.0,20.0,30.0);
list.stream().map(x->x+x*0.05).forEach(x-> System.out.println(x));
}
}
3、用lambda表达式实现map与reduce
既然提到了map,又怎能不提到reduce。reduce与map一样,也是函数式编程里最重要的几个方法之一。。。map的作用是将一个对象变为另外一个,而reduce实现的则是将所有值合并为一个,请看:
package OSChina.Lambda;
import java.util.Arrays;
import java.util.List;
public class Test3 {
public static void main(String[] args) {
//before java8
List<Double> cost = Arrays.asList(10.0, 20.0,30.0);
double sum = 0;
for(double each:cost) {
each += each * 0.05;
sum += each;
}
System.out.println("before java8,"+sum);
//after java8
List<Double> list = Arrays.asList(10.0,20.0,30.0);
double sum2 = list.stream().map(x->x+x*0.05).reduce((sum1,x)->sum1+x).get();
System.out.println("after java8,"+sum2);
}
}
相信用map+reduce+lambda表达式的写法高出不止一个level。
4、filter操作
filter也是我们经常使用的一个操作。在操作集合的时候,经常需要从原始的集合中过滤掉一部分元素。
package OSChina.Lambda;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
public class Test3 {
public static void main(String[] args) {
List<Double> cost = Arrays.asList(10.0, 20.0,30.0,40.0);
List<Double> filteredCost = cost.stream().filter(x -> x > 25.0).collect(Collectors.toList());
filteredCost.forEach(x -> System.out.println(x));
}
}
5、与函数式接口Predicate配合
除了在语言层面支持函数式编程风格,Java 8也添加了一个包,叫做 java.util.function。它包含了很多类,用来支持Java的函数式编程。其中一个便是Predicate,使用 java.util.function.Predicate 函数式接口以及lambda表达式,可以向API方法添加逻辑,用更少的代码支持更多的动态行为。Predicate接口非常适用于做过滤。
package OSChina.Lambda;
import java.lang.reflect.Array;
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
public class Test4 {
public static void filterTest(List<String> languages, Predicate<String> condition) {
languages.stream().filter(x -> condition.test(x)).forEach(x -> System.out.println(x + " "));
}
public static void main(String[] args) {
List<String> languages = Arrays.asList("Java","Python","scala","Shell","R");
filterTest(languages,x->x.startsWith("J"));//Java
filterTest(languages,x -> x.endsWith("a"));//Java,scala
filterTest(languages,x -> true);//Java,Python,scala,Shell,R
filterTest(languages,x -> false);//
filterTest(languages,x -> x.length() > 4);//Python,scala,Shell,
}
}
浅谈lambda表达式<最通俗易懂的讲解的更多相关文章
- 浅谈Lambda表达式详解
lambda简介 lambda运算符:所有的lambda表达式都是用新的lambda运算符 " => ",可以叫他,“转到”或者 “成为”.运算符将表达式分为两部分,左边指定 ...
- 浅谈-Lambda
Lambda简化了匿名委托的使用,让你让代码更加简洁,优雅.据说它是微软自c#1.0后新增的最重要的功能之一. 简介: lambda运算符:所有的lambda表达式都是用新的lambda运算符 &qu ...
- 浅入浅出Lambda表达式
大家在开发中会经常看到也会经常使用lambda表达式. 园子里也有很多详解lambda表达式的文章,多是从横向来讲述. 但lambda表达式到底如何变成现在这个样子,表达式的形式到底代表什么含义,这些 ...
- 浅谈OGNL表达式
OGNL(Object-Graph Navigation Language):对象视图导航语言 ${user.addr.name}这样的写法就叫对象视图导航 OGNL不仅可以视图导航,支持EL表达式更 ...
- Lambda表达式演变
Lambda表达式是一种匿名函数. 演变步骤: 一般的方法委托 => 匿名函数委托 => Lambda表达式 Lambda表达式其实并不陌生,他的前生就是匿名函数,所以要谈La ...
- Linq快速入门——Lambda表达式的前世今生
Linq快速入门——Lambda表达式的前世今生 Lambda表达式其实并不陌生,他的前生就是匿名函数,所以要谈Lambda表达式,就不得不谈匿名函数,要谈匿名函数,那又要不得不谈委托. 何为委托 ...
- Java8 Lambda表达式详解手册及实例
先贩卖一下焦虑,Java8发于2014年3月18日,距离现在已经快6年了,如果你对Java8的新特性还没有应用,甚至还一无所知,那你真得关注公众号"程序新视界",好好系列的学习一下 ...
- 浅谈DFS序
浅谈DFS序 本篇随笔简要讲解一下信息学奥林匹克竞赛中有关树的DFS序的相关内容. DFS序的概念 先来上张图: 树的DFS序,简单来讲就是对树从根开始进行深搜,按搜到的时间顺序把所有节点打上时间戳. ...
- 函数式编程/lambda表达式入门
函数式编程/lambda表达式入门 本篇主要讲解 lambda表达式的入门,涉及为什么使用函数式编程,以及jdk8提供的函数式接口 和 接口的默认方法 等等 1.什么是命令式编程 命令式编程就是我们去 ...
随机推荐
- 修改VScode底部状态栏颜色
点击齿轮进入setting 搜索workbench.colorCustomizations,然后点击编辑setting.json 修改为你喜欢的颜色即可,我这里修改为蓝色 例子: "work ...
- 【vue-01】快速入门
什么是vue vue是渐进式的JavaScript框架 什么是渐进式? 你可以在原有大系统的上面,把一两个组件改用vue实现,:也可以整个项目用vue全家桶开发. vue是一个轻量级的MVVM ...
- RabbitMQ实现延时消息的两种方法
目录 RabbitMQ实现延时消息的两种方法 1.死信队列 1.1消息什么时候变为死信(dead-letter) 1.2死信队列的原理 1.3 代码实现 1.4死信队列的一个小坑 2 .延时插件 2. ...
- 【js】Leetcode每日一题-二叉树的堂兄弟节点
[js]Leetcode每日一题-二叉树的堂兄弟节点 [题目描述] 在二叉树中,根节点位于深度 0 处,每个深度为 k 的节点的子节点位于深度 k+1 处. 如果二叉树的两个节点深度相同,但 父节点不 ...
- SwiftUI 简明教程之属性包装器
本文为 Eul 样章,如果您喜欢,请移步 AppStore/Eul 查看更多内容. Eul 是一款 SwiftUI & Combine 教程 App(iOS.macOS),以文章(文字.图片. ...
- Envoy:经过envoy代理后获取客户端真实IP
在envoy作为前端代理时,用户ip的获取很重要,一般获取ip的方式.都是通过Header中的 X-Forward-For. X-Real-IP或 Remote addr 等属性获取,但是如果确保En ...
- 变体 variety 计算机学科中的改变类型;输入法的 类型
变体_百度百科 中文为改变原来的体式.或者计算机学科中的改变类型. 变体 variety 输入法的 类型
- 要想在for语句中直接定义一个变量
要想在for语句中直接 定义一个变量 (如下的代码) 1 for(uint16_t i=0;i<10;i++); 2 if( GPIO_ReadInputDataBit(GPIOA, GPI ...
- Linux服务之DNS服务篇
一.DNS服务概述 DNS(Domain Name System)域名系统,在TCP/IP 网络中有非常重要的地位,能够提供域名与IP地址的解析服务. DNS 是一个分布式数据库,命名系统采用层次的逻 ...
- UCOS明白解析
UCOSII 是一个可以基于 ROM 运行的.可裁减的.抢占式.实时多任务内核,具有高度可移植性,特别适合于微处理器和控制器,是和很多商业操作系统性能相当的实时操作系统(RTOS).为了提供最 ...