Java中lambda表达式学习
一.Lambda表达式的基础语法:
Java8中引入了一个新的操作符"->"该操作符称为箭头操作符或Lambda操作符,箭头操作符将Lambda表达式拆分为两部分:
左侧:Lambda表达式的参数列表
右侧:Lambda表达式所需要执行的功能,即Lambda体
语法一:无参数,无返回值
() -> System.out.println("hello");
语法二:有一个参数,无返回值
(x) -> System.out.println("hello" + x);
语法三:只有一个参数,无返回值
x -> System.out.println("hello" + x);
语法四:有两个以上参数,又返回值,并且Lambda体中有多条语句
Comparator<Interger> com = (x, y) -> {
System.out.println("hello");
return Interger.compare(x, y);
};
语法五:如果Lambda体只有一条语句,return和大括号都可以省略
Comparator<Interger> com = (x, y) -> Interger.compare(x, y);
语法六:Lambda表达式参数列表的数据类型可以省略不写,JVM编译器通过上下文推断出数据类型(“类型推断”)
Comparator<Interger> com = (Interger x,Interger y) -> Interger.compare(x, y);
二.使用Lambda优化程序
接口->外部实现类->静态内部类->局部内部类->匿名内部类->lambda->方法引用方式
/**
* 定义一个函数式接口:
* 函数式接口是一个接口中只有一个非默认且非Object类的方法
*/
@FunctionalInterface
interface ILike {
//默认都为抽象方法(一般都省略abstract)
void lambda(); default void print() {
System.out.println("我是默认方法");
} String toString(); boolean equals(Object obj);
} //外部实现类
class Like1 implements ILike { @Override
public void lambda() {
System.out.println("外部实现类");
}
} public class TestLambda { public static void printLambda() {
System.out.println("我是TestLambda类的静态方法");
} //静态内部类
static class Like2 implements ILike { @Override
public void lambda() {
System.out.println("静态内部类");
}
} public static void main(String[] args) { ILike like = new Like1();
like.lambda(); like = new Like2();
like.lambda(); //局部内部类
class Like3 implements ILike { @Override
public void lambda() {
System.out.println("局部内部类");
}
} like = new Like3();
like.lambda(); //匿名内部类,没有类的名字,必须借助接口或者父类实现
like = new ILike() {
@Override
public void lambda() {
System.out.println("匿名内部类");
}
}; like.lambda(); //lambda表达式
like = () -> System.out.println("lambda表达式");
like.lambda(); //方法引用方式(类::静态方法)
like = TestLambda::printLambda;
like.lambda(); } }
三.Lambda表达式的延迟执行
目的:避免性能浪费
字符串是否需要拼接用例:
- 普通方法:如果level不符合要求依旧会拼接字符串造成性能浪费
- lambda表达式:level不符合要求不会拼接,减少性能浪费
- 使用供给型接口优化lambda
public class demo02Logger { public static void showLog1(int level, String message) {
if (level == 1) {
System.out.println("showLog1:level1:" + message);
}
} public static void showLog2(int level, MessageBuilder mb) {
if (level == 1) {
System.out.println("showLog2:level1:" + mb.builderMessage());
}
} public static void showLog3(int level, Supplier<String> stringSupplier) {
if (level == 1) {
System.out.println("showLog3:level1:" + stringSupplier.get());
}
} public static void main(String[] args) {
String msg1 = "msg1";
String msg2 = "msg2";
String msg3 = "msg3"; //普通方法:如果level不符合要求依旧会拼接字符串造成性能浪费
showLog1(2, msg1 + msg2 + msg3); //lambda表达式:level不符合要求不会拼接,减少性能浪费
showLog2(1, () -> {
System.out.println("showLog2:level为1才执行拼接");
return msg1 + msg2 + msg3;
}); showLog2(2, () -> {
System.out.println("showLog2:level为2不执行拼接");
return msg1 + msg2 + msg3;
}); //使用供给型函数式接口
showLog3(1,() -> {
System.out.println("showLog3:level为1才执行拼接");
return msg1 + msg2 + msg3;
}); showLog3(2,() -> {
System.out.println("showLog3:level为2不执行拼接");
return msg1 + msg2 + msg3;
});
}
} @FunctionalInterface
interface MessageBuilder {
String builderMessage();
}
结果:
使用lambda优化了性能
showLog2:level为1才执行
showLog2:level1:msg1msg2msg3
showLog3:level为1才执行
showLog3:level1:msg1msg2msg3
四.Lambda表达式需要“函数式接口”的支持
函数式接口:接口中只有一个抽象的方法,称之为函数式接口。可以使用注解 @FunctionalInterface 修饰(用于检查是否是函数式接口)
Java8 内置的四大核心函数式接口:(还有很多子接口)
- Consumer<T>:消费型接口
- void accept(T t);
- Supplier<T>:供给型接口
- T get();
- R apply(T t);
- Predicate<T>:断言型接口
- boolean test(T t)
Function<T,R>:函数型接口(将T类型转换为R类型)
1.方法引用:若Lambda体中的内容有方法已经实现了,我们可以使用“方法引用”(理解为Lambda的另一种表现形式)
主要有三种语法格式:(所引用的方法参数和返回值要与函数式接口的抽象方法参数和返回值一致)
- 对象::实例方法
- 类::静态方法
- 类::实例方法(必须是第一个参数为实例方法,调用者第二个参数为实例方法的参数)
@Test
public void test() { //lambda
PrintStream ps = System.out;
Consumer<String> con1 = (x) -> ps.println(x);
con1.accept("con1"); //对象::实例方法名(返回值和参数需要与方法调用者一致)
Consumer<String> con2 = System.out::println;
con2.accept("con2"); //lambda
Comparator<Integer> com1 = (x, y) -> Integer.compare(x, y);
System.out.println(com1.compare(5, 6)); //类::静态方法名(返回值和参数需要与方法调用者一致)
Comparator<Integer> com2 = Integer::compare;
System.out.println(com2.compare(5, 6)); //lambda
BiPredicate<String, String> bp1 = (x, y) -> x.equals(y);
System.out.println(bp1.test("aaa","aaa")); //类::实例方法名(条件是第一个参数为实例方法调用者,第二个参数为实例方法的参数)
// (返回值和参数需要与方法调用者一致)
BiPredicate<String,String> bp2 = String::equals;
System.out.println(bp2.test("aaa","aaa")); }
2.构造器引用:
格式:ClassName::new(所调用的构造器参数要与函数式接口的抽象方法参数一致)
class Mytest{ private Integer id;
private String testmes; public Mytest() {
} public Mytest(Integer id) {
this.id = id;
} public Mytest(Integer id, String testmes) {
this.id = id;
this.testmes = testmes;
} @Override
public String toString() {
return "Mytest{" +
"id=" + id +
", testmes='" + testmes + '\'' +
'}';
}
} @Test
public void test2(){ //Lambda(调用无参构造函数)
Supplier<Mytest> sup1 = ()->new Mytest();
System.out.println(sup1.get()); //构造器引用方式(调用无参构造函数)
Supplier<Mytest> sup2 = Mytest::new;
System.out.println(sup2.get()); //Lambda(调用一个参数和一个返回值构造函数)
Function<Integer,Mytest> fun1 = (x)->new Mytest(x);
System.out.println(fun1.apply(3)); //构造器引用方式(调用一个参数和一个返回值构造函数)
Function<Integer,Mytest> fun2 = Mytest::new;
System.out.println(fun2.apply(3)); //Lambda(调用两个参数和一个返回值构造函数)
BiFunction<Integer,String,Mytest> bi1 = (x,y)->new Mytest(x,y);
System.out.println(bi1.apply(3,"mytest3")); //构造器引用方式(调用两个参数和一个返回值构造函数)
BiFunction<Integer,String,Mytest> bi2 = Mytest::new;
System.out.println(bi2.apply(3,"mytest3")); }
3.数组引用
格式:Type::new
@Test
public void testarray(){ //lambda
Function<Integer,String[]> fun1 = (x) -> new String[x];
System.out.println(fun1.apply(10).length); //10 //数组引用
Function<Integer,String[]> fun2 = String[]::new;
System.out.println(fun2.apply(15).length); //15 }
Java中lambda表达式学习的更多相关文章
- Java中lambda表达式详解
原文地址:http://blog.laofu.online/2018/04/20/java-lambda/ 为什么使用lambda 在java中我们很容易将一个变量赋值,比如int a =0;int ...
- Java 8 Lambda表达式学习和理解
Java 8 Lambda表达式和理解 说明:部分资料来源于网络 时间:20190704 Lambda 表达式,也可称为闭包,它是推动 Java 8 发布的最重要新特性.Lambda 允许把函数作为一 ...
- java 8 中lambda表达式学习
转自 http://blog.csdn.net/renfufei/article/details/24600507 http://www.jdon.com/idea/java/10-example-o ...
- Java中Lambda表达式的使用
简介(译者注:虽然看着很先进,其实Lambda表达式的本质只是一个"语法糖",由编译器推断并帮你转换包装为常规的代码,因此你可以使用更少的代码来实现同样的功能.本人建议不要乱用,因 ...
- Java中Lambda表达式的使用(转)
https://www.cnblogs.com/franson-2016/p/5593080.html 简介(译者注:虽然看着很先进,其实Lambda表达式的本质只是一个"语法糖" ...
- 快速掌握Java中Lambda表达式的用法
Lambda表达式的作用: Lambda表达式的作用就是简化代码开发,让代码看起来更加简介.它是用来简化匿名内部类的.但是并不是所有的匿名内部类都能用Lambda表达式简化,Lambda表达式是有使用 ...
- Java中Lambda表达式的简单使用
Lambda表达式是Java SE 8中一个重要的新特性.你可以把 Lambda表达式 理解为是一段可以传递的代码 (将代码像数据一样进行传递).可以写出更简洁.更灵活的代码.作为一种更紧凑的代码风格 ...
- Java中lambda(λ)表达式的语法
举一个排序的例子,我们传入代码来检查一个字符串是否比另一个字符串短.这里要计算: first.length() - second.length() first和second是什么?他们都是字符串.Ja ...
- 为什么Java中lambda表达式不能改变外部变量的值,也不能定义自己的同名的本地变量呢?
作者:blindpirate链接:https://www.zhihu.com/question/361639494/answer/948286842来源:知乎著作权归作者所有.商业转载请联系作者获得授 ...
随机推荐
- 物联网通讯协议:MQTT,CoAP,NB-IOT,RFID,BLUETOOTH,NFC
一.按网络四层协议分类: NB-IoT,LORA,WIFI,蓝牙,zigbee,4G都是物理层的,这几个都需要芯片模组支持(硬件支持) 而MQTT,COAP,HTTP都是应用层协议,这些需要开发服务器 ...
- Mysql系列(二)—— Mysql支持的数据类型
Mysql版本众多,每个版本支持的数据类型繁多且不一样,本篇文章中主要基于MySQL Community Server 5.7.22介绍常用的数据类型,包括其特点以及区别. 一.数据类型 正确的定义表 ...
- [shell] while read line 与for循环的区别
[shell] while read line 与for循环的区别 while read line 与for循环的区别---转载整理 while read line 是一次性将文件信息读入并赋值给变量 ...
- Spring Web MVC框架简介
Web MVC framework框架 Spring Web MVC框架简介 Spring MVC的核心是`DispatcherServlet`,该类作用非常多,分发请求处理,配置处理器映射,处理视图 ...
- springboot整合docker部署
环境安装 首先,需要安装Docker(例如:docker for windows) 下载地址:https://download.docker.com/win/stable/Docker%20for%2 ...
- Java 流程控制 之 分支结构——条件判断语句
一.判断语句 1.判断语句1-- 单 if 语句(单分支结构) 语法格式: if(条件表达式){ 语句体; } 执行流程: 首先判断条件表达式看其结果是 true 还是 false: 如果是 tru ...
- 如何使用Python的Django框架创建自己的网站
如何使用Python的Django框架创建自己的网站 Django建站主要分四步:1.创建Django项目,2.将网页模板移植到Django项目中,3.数据交互,4.数据库 1创建Django项目 本 ...
- kbmmw 中的Boyer-Moore算法
kbmmw 5.10 版本中实现了一个非常好用的字符串搜索算法,即Boyer-Moore算法. 在用于查找子字符串的算法当中,BM(Boyer-Moore)算法是目前被认为最高效的字符串搜索算法, 它 ...
- ElasticSearch(十三):Spring Data ElasticSearch 的使用(三)——NativeSearchQuery 高亮查询
在Elasticsearch的实际应用中,经常需要将匹配到的结果字符进行高亮显示,此处采取NativeSearchQuery原生查询的方法,实现查询结果的高亮显示. /** * 高亮查询 */ @Te ...
- 纯数据结构Java实现(8/11)(Trie)
欢迎访问我的自建博客: CH-YK Blog.