浅谈lambda表达式<最通俗易懂的讲解>

本文介绍了Java8的重要新特性——lambda表达式,它简化了代码,替代了匿名内部类,使得行为传递更加直观。文章详细讲解了lambda的用法,包括替代匿名内部类、四大内置函数式接口如Consumer、Supplier、Function和Predicate,以及方法引用和构造器引用。此外,还展示了lambda在集合操作、map、reduce、filter和Predicate配合中的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Java8发布已经有一段时间了,这次发布的改动比较大,很多人将这次改动与Java5的升级相提并论。Java8其中一个很重要的新特性就是lambda表达式,允许我们将行为传到函数中。想想看,在Java8 之前我们想要将行为传入函数,仅有的选择就是匿名内部类。Java8发布以后,lambda表达式将大量替代匿名内部类的使用,简化代码的同时,更突出了原来匿名内部类中最重要的那部分包含真正逻辑的代码。尤其是对于做数据的同学来说,当习惯使用类似scala之类的函数式编程语言以后,体会将更加深刻。现在我们就来看看Java8中lambda表达式的一些常见写法。

lambda体中调用方法的参数列表和返回值类型,要和函数式接口中抽象方法的参数列表和返回值类型保持一致。

一、替代匿名内部类

lambda表达式用的最多的场合就是替代匿名内部类,实现Runnable接口是匿名内部类的经典例子。lambda表达式的功能相当强大,用()->就可以代替整个匿名内部类!

packageOSChina.Lambda;

importorg.junit.Test;

importjava.util.Comparator;
importjava.util.function.Consumer;
importjava.util.function.Function;
importjava.util.function.Predicate;
importjava.util.function.Supplier;

publicclassTest1{
publicstaticvoidmain(String[] args){
Runnable runnable =newRunnable() {
@Override
publicvoidrun(){
System.out.println("普通,线程启动");
}
};
runnable.run();
test2();
test3();
test4();
test5();
}

//无参数,无返回值
publicstaticvoidtest2(){
//“->”左边只有一个小括号,表示无参数,右边是Lambda体(就相当于实现了匿名内部类里面的方法了,(即就是一个可用的接口实现类了。))
Runnable runnable = ()->System.out.println("Lambda 表达式方式,线程启动");
runnable.run();
}

//有一个参数,并且无返回值
publicstaticvoidtest3(){
//这个e就代表所实现的接口的方法的参数,
Consumer<String> consumer = e->System.out.println("Lambda 表达式方式,"+e);
consumer.accept("传入参数");
}

//有两个以上的参数,有返回值,并且 Lambda 体中有多条语句
publicstaticvoidtest4(){
//Lambda 体中有多条语句,记得要用大括号括起来
Comparator<Integer> com = (x, y) -> {
System.out.println("函数式接口");
returnInteger.compare(x, y);
};
intcompare = com.compare(100,244);
System.out.println("有两个以上的参数,有返回值,"+compare);
}

//若 Lambda 体中只有一条语句, return 和 大括号都可以省略不写
publicstaticvoidtest5(){
//Comparator com = (x, y) -> Integer.compare(100, 244);
System.out.println("若 Lambda 体中只有一条语句, return 和 大括号都可以省略不写,"+Integer.compare(100,244));
}
}

浅谈lambda表达式<最通俗易懂的讲解>

 

二、Java8四大内置函数式接口

如果使用lambda还要自己写一个接口的话太麻烦,所以Java自己提供了一些接口:

1、Consumer 消费性接口:void accept(T t);

//有一个参数,并且无返回值
publicstaticvoidtest3(){
//这个e就代表所实现的接口的方法的参数,
Consumer<String> consumer = e->System.out.println("Lambda 表达式方式,"+e);
consumer.accept("传入参数");
}

2、Supplier供给型接口:T get();

package OSChina.Lambda;

importjava.util.ArrayList;
importjava.util.function.Supplier;

publicclassTest2{
publicstaticvoidmain(String[] args){
ArrayList<Integer> res = getNumList(10,()->(int)(Math.random()*100));
System.out.println(res);
}

publicstaticArrayList<Integer> getNumList(intnum, Supplier<Integer> sup){
ArrayList<Integer>list=newArrayList<>();
for(inti =0; i < num; i++) {
Integer e = sup.get();
list.add(e);
}
returnlist;
}
}

浅谈lambda表达式<最通俗易懂的讲解>

 

3、Function 函数式接口:R apply(T t);

package OSChina.Lambda;

importjava.util.function.Function;

publicclassTest2 {
publicstaticvoidmain(String[] args) {
StringnewStr = strHandler("abc",(str)->str.toUpperCase());
System.out.println(newStr);
newStr = strHandler(" abc ",(str)->str.trim());
System.out.println(newStr);
}

publicstaticStringstrHandler(Stringstr,Function<String,String>fun){
returnfun.apply(str);
}
}

浅谈lambda表达式<最通俗易懂的讲解>

 

4、Predicate 断言式接口:boolean test(T t);

判断一些字符串数组判断长度>2的字符串:

package OSChina.Lambda;

importjava.util.ArrayList;
importjava.util.Arrays;
importjava.util.List;
importjava.util.function.Predicate;

publicclassTest2 {
publicstaticvoidmain(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);
}

publicstaticList<String> filterStr(List<String> list, Predicate<String> pre){
ArrayList<String> arrayList =newArrayList<>();
for(Stringstr:list){
if(pre.test(str)) {
arrayList.add(str);
}
}
returnarrayList;
}
}

浅谈lambda表达式<最通俗易懂的讲解>

 

三、方法引用与构造器引用

要求:实现抽象方法的参数列表和返回值类型,必须与方法引用的方法的参数列表和返回值类型保持一致!

方法引用:使用操作符“::”将类与方法分隔开来。

对象::实例方法名
类::静态方法名
类::实例方法名

举个例子:

publicstaticvoidtest9(){
Comparator<Integer> comparator = (x,y)->Integer.compare(x,y);
Comparator<Integer> comparator1 = Integer::compare;
intcompare = comparator.compare(1,2);
intcompare1 = comparator1.compare(1,2);
System.out.println("compare:"+compare);
System.out.println("compare1:"+compare1);
}

浅谈lambda表达式<最通俗易懂的讲解>

 

四、lambda表达式的一些常见用法

1、使用lambda表达式对集合进行迭代

package OSChina.Lambda;

importjava.util.Arrays;
importjava.util.List;

publicclassTest3{
publicstaticvoidmain(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));
}
}

浅谈lambda表达式<最通俗易懂的讲解>

 

2、用lambda表达式实现map

map函数可以说是函数式编程里最重要的一个方法了。map的作用是将一个对象变换为另外一个。在我们的例子中,就是通过map方法将cost增加了0,05倍的大小然后输出。

package OSChina.Lambda;

importjava.util.Arrays;
importjava.util.List;

publicclassTest3{
publicstaticvoidmain(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));
}
}

浅谈lambda表达式<最通俗易懂的讲解>

 

3、用lambda表达式实现map与reduce

既然提到了map,又怎能不提到reduce。reduce与map一样,也是函数式编程里最重要的几个方法之一。。。map的作用是将一个对象变为另外一个,而reduce实现的则是将所有值合并为一个,请看:

package OSChina.Lambda;

importjava.util.Arrays;
importjava.util.List;

publicclassTest3{
publicstaticvoidmain(String[] args){
//before java8
List<Double> cost = Arrays.asList(10.0,20.0,30.0);
doublesum =0;
for(doubleeach: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);
doublesum2 =list.stream().map(x->x+x*0.05).reduce((sum1,x)->sum1+x).get();
System.out.println("after java8,"+sum2);
}
}

浅谈lambda表达式<最通俗易懂的讲解>

 

相信用map+reduce+lambda表达式的写法高出不止一个level。

4、filter操作

filter也是我们经常使用的一个操作。在操作集合的时候,经常需要从原始的集合中过滤掉一部分元素。

packageOSChina.Lambda;

importjava.util.Arrays;
importjava.util.List;
importjava.util.stream.Collectors;

publicclassTest3{
publicstaticvoidmain(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));
}
}

浅谈lambda表达式<最通俗易懂的讲解>

 

5、与函数式接口Predicate配合

除了在语言层面支持函数式编程风格,Java 8也添加了一个包,叫做 java.util.function。它包含了很多类,用来支持Java的函数式编程。其中一个便是Predicate,使用 java.util.function.Predicate 函数式接口以及lambda表达式,可以向API方法添加逻辑,用更少的代码支持更多的动态行为。Predicate接口非常适用于做过滤。

package OSChina.Lambda;

importjava.lang.reflect.Array;
importjava.util.Arrays;
importjava.util.List;
importjava.util.function.Predicate;

publicclassTest4 {
publicstaticvoidfilterTest(List<String> languages, Predicate<String> condition) {
languages.stream().filter(x -> condition.test(x)).forEach(x -> System.out.println(x +" "));
}

publicstaticvoidmain(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,
}
}
&lt;think&gt;好的,我现在需要帮助用户了解MyBatis-Plus中LambdaQueryWrapper的.lt()方法的使用。用户之前询问过“lt是大于还是小于”,现在转向具体的框架方法应用。首先,我得回顾之前的回答,确保信息一致。用户提供的引用提到LambdaQueryWrapper通过Lambda表达式构建查询条件,但用户现在具体问的是.lt()方法。我需要解释.lt()的作用,即“小于”条件构造,并给出示例。首先,要确认.lt()在MyBatis-Plus中的功能。根据文档,lt方法用于添加字段小于指定值的条件。结合Lambda表达式,可以避免硬编码字段名,提高代码可读性。接下来,需要构造一个基础示例。比如,查询年龄小于30的用户。代码结构应包括创建LambdaQueryWrapper实例,调用.lt()方法,并链式调用其他方法如.select()或.list()。要注意代码的正确性,比如方法参数是否正确。然后,可能需要展示多条件组合的示例,比如年龄小于30且状态为启用。这里需要说明如何链式调用.lt()和.eq()方法,强调LambdaQueryWrapper的链式特性。用户可能对动态条件构造感兴趣,比如根据变量判断是否添加条件。这时候可以用.lt()配合条件判断,如if语句,动态构建查询条件。这也是实际开发中常见的场景,需要举例说明。此外,结合其他比较方法,如.le()(小于等于)、.gt()(大于)等,可以进一步扩展说明,帮助用户理解整个条件构造体系。但需注意不要偏离用户当前的问题,保持重点在.lt()。需要检查是否需要提到与数据库字段的映射,确保Lambda表达式引用的实体类属性正确。例如,User::getAge对应数据库中的age字段,避免字段名错误导致的查询问题。可能还需要提醒用户关于参数类型的注意事项,比如字段类型必须与比较值类型一致,否则可能抛出异常。例如,如果数据库字段是整数,而传入字符串,会导致错误。后,生成相关问题,帮助用户深入理解,比如动态条件构造、多条件组合、与其他方法比较等。这些问题应该基于常见的后续疑问,引导用户进一步学习。总结回答结构:先概述.lt()的作用,再分点说明不同使用场景的示例,强调Lambda表达式的好处,后注意事项和相关问题。确保代码示例正确,格式符合用户要求,如使用正确的语法和引用标记。&lt;/think&gt;### LambdaQueryWrapper `.lt()` 方法使用说明 在 MyBatis-Plus 中,`.lt()` 是 `LambdaQueryWrapper` 的条件构造方法之一,用于生成 **字段小于指定值** 的查询条件(即 SQL 中的 `&lt;` 运算符)。结合 Lambda 表达式,可以直接引用实体类的属性,避免硬编码字段名,提高代码可维护性[^1][^2]。 --- ### 一、基础用法 #### 语法格式 ```java queryWrapper.lt(实体类字段的Lambda表达式, 比较值) ``` #### 示例场景 假设有 `User` 实体类,包含字段 `age`,查询年龄小于 30 的用户: ```java LambdaQueryWrapper&lt;User&gt; queryWrapper = new LambdaQueryWrapper&lt;&gt;(); queryWrapper.lt(User::getAge, 30); // 等价于 SQL: age &lt; 30 List&lt;User&gt; userList = userMapper.selectList(queryWrapper); ``` --- ### 二、多条件组合 `.lt()` 可与其他条件方法(如 `.eq()`、`.gt()`)链式调用,实现复杂查询。 #### 示例 查询年龄小于 30 **且** 状态为 "启用" 的用户: ```java queryWrapper.lt(User::getAge, 30) .eq(User::getStatus, "启用"); // SQL: WHERE age &lt; 30 AND status = '启用' ``` --- ### 三、动态条件构造 通过条件判断灵活控制查询逻辑: ```java Integer maxAge = 30; // 动态传入的阈值 LambdaQueryWrapper&lt;User&gt; queryWrapper = new LambdaQueryWrapper&lt;&gt;(); if (maxAge != null) { queryWrapper.lt(User::getAge, maxAge); } List&lt;User&gt; userList = userMapper.selectList(queryWrapper); ``` --- ### 四、与其他比较方法对比 | 方法 | 运算符 | 示例 | SQL 等价 | |-----------|--------|----------------------|---------------------| | `.lt()` | &lt; | `lt(User::getAge,30)`| `age &lt; 30` | | `.le()` | &lt;= | `le(User::getAge,30)`| `age &lt;= 30` | | `.gt()` | &gt; | `gt(User::getAge,30)`| `age &gt; 30` | | `.ge()` | &gt;= | `ge(User::getAge,30)`| `age &gt;= 30` | --- ### 五、注意事项 1. **字段类型匹配**:比较值的类型需与数据库字段类型一致(如 `Integer` vs `INT`); 2. **空值处理**:若比较值为 `null`,可能引发异常,建议提前判空; 3. **链式调用顺序**:多个条件默认以 `AND` 连接,需注意逻辑关系。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值