Lambda表达式你到哪个境界了?

简介: 日常开发中,我们很多时候需要用到Java 8的Lambda表达式,它允许把函数作为一个方法的参数,让我们的代码更优雅、更简洁。所以整理了一波工作中,我常用的,有哪些Lambda表达式。看完一定会有帮助的。

1.list转map

工作中,我们经常遇到list转map的案例。Collectors.toMap就可以把一个list数组转成一个Map。代码如下:

public class TestLambda {
    public static void main(String[] args) {
        List<UserInfo> userInfoList = new ArrayList<>();
        userInfoList.add(new UserInfo(1L, "小男孩", 18));
        userInfoList.add(new UserInfo(2L, "程序员", 27));
        userInfoList.add(new UserInfo(2L, "捡瓶子", 26));
        /**
         *  list 转 map
         *  使用Collectors.toMap的时候,如果有可以重复会报错,所以需要加(k1, k2) -> k1
         *  (k1, k2) -> k1 表示,如果有重复的key,则保留第一个,舍弃第二个
         */
        Map<Long, UserInfo> userInfoMap = userInfoList.stream().collect(Collectors.toMap(UserInfo::getUserId, userInfo -> userInfo, (k1, k2) -> k1));
        userInfoMap.values().forEach(a->System.out.println(a.getUserName()));
    }
}
//运行结果
小男孩
程序员

类似的,还有Collectors.toList()、Collectors.toSet(),表示把对应的流转化为list或者Set。

2.filter()过滤

从数组集合中,过滤掉不符合条件的元素,留下符合条件的元素。

List<UserInfo> userInfoList = new ArrayList<>();
userInfoList.add(new UserInfo(1L, "捡破烂", 18));
userInfoList.add(new UserInfo(2L, "程序员", 27));
userInfoList.add(new UserInfo(3L, "小男孩", 26));
/**
 * filter 过滤,留下超过18岁的用户
 */
List<UserInfo> userInfoResultList = userInfoList.stream().filter(user -> user.getAge() > 18).collect(Collectors.toList());
userInfoResultList.forEach(a -> System.out.println(a.getUserName()));
//运行结果
程序员
小男孩

3.foreach遍历

foreach 遍历list,遍历map,真的很丝滑。

/**
 * forEach 遍历集合List列表
 */
List<String> userNameList = Arrays.asList("小男孩", "程序员", "捡瓶子");
userNameList.forEach(System.out::println);
HashMap<String, String> hashMap = new HashMap<>();
hashMap.put("公众号", "小男孩");
hashMap.put("职业", "程序员");
hashMap.put("昵称", "捡瓶子");
/**
 *  forEach 遍历集合Map
 */
hashMap.forEach((k, v) -> System.out.println(k + ":\t" + v));
//运行结果
小男孩
程序员
捡瓶子
职业: 程序员
公众号: 小男孩
昵称: 捡瓶子

4.groupingBy分组

提到分组,相信大家都会想起SQL的group by。我们经常需要一个List做分组操作。比如,按城市分组用户。在Java8之前,是这么实现的:

List<UserInfo> originUserInfoList = new ArrayList<>();
originUserInfoList.add(new UserInfo(1L, "小男孩", 18,"深圳"));
originUserInfoList.add(new UserInfo(3L, "捡瓶子", 26,"湛江"));
originUserInfoList.add(new UserInfo(2L, "程序员", 27,"深圳"));
Map<String, List<UserInfo>> result = new HashMap<>();
for (UserInfo userInfo : originUserInfoList) {
  String city = userInfo.getCity();
  List<UserInfo> userInfos = result.get(city);
  if (userInfos == null) {
      userInfos = new ArrayList<>();
      result.put(city, userInfos);
    }
  userInfos.add(userInfo);
}

而使用Java8的groupingBy分组器,清爽无比:

Map<String, List<UserInfo>> result = originUserInfoList.stream()
.collect(Collectors.groupingBy(UserInfo::getCity));

5. sorted+Comparator 排序

工作中,排序的需求比较多,使用sorted+Comparator排序,真的很香。

List<UserInfo> userInfoList = new ArrayList<>();
userInfoList.add(new UserInfo(1L, "小男孩", 18));
userInfoList.add(new UserInfo(3L, "捡瓶子", 26));
userInfoList.add(new UserInfo(2L, "程序员", 27));
/**
 *  sorted + Comparator.comparing 排序列表,
 */
userInfoList = userInfoList.stream().sorted(Comparator.comparing(UserInfo::getAge)).collect(Collectors.toList());
userInfoList.forEach(a -> System.out.println(a.toString()));
System.out.println("开始降序排序");
/**
 * 如果想降序排序,则可以使用加reversed()
 */
userInfoList = userInfoList.stream().sorted(Comparator.comparing(UserInfo::getAge).reversed()).collect(Collectors.toList());
userInfoList.forEach(a -> System.out.println(a.toString()));
//运行结果
UserInfo{userId=1, userName='小男孩', age=18}
UserInfo{userId=3, userName='捡瓶子', age=26}
UserInfo{userId=2, userName='程序员', age=27}
开始降序排序
UserInfo{userId=2, userName='程序员', age=27}
UserInfo{userId=3, userName='捡瓶子', age=26}
UserInfo{userId=1, userName='小男孩', age=18}

6.distinct去重

distinct可以去除重复的元素:

List<String> list = Arrays.asList("A", "B", "F", "A", "C");
List<String> temp = list.stream().distinct().collect(Collectors.toList());
temp.forEach(System.out::println);


7. findFirst 返回第一个

findFirst 很多业务场景,我们只需要返回集合的第一个元素即可:

List<String> list = Arrays.asList("A", "B", "F", "A", "C");
list.stream().findFirst().ifPresent(System.out::println);

8.anyMatch是否至少匹配一个元素

anyMatch 检查流是否包含至少一个满足给定谓词的元素。

Stream<String> stream = Stream.of("A", "B", "C", "D");
boolean match = stream.anyMatch(s -> s.contains("C"));
System.out.println(match);
//输出
true

9. allMatch 匹配所有元素

  1. allMatch 检查流是否所有都满足给定谓词的元素。
Stream<String> stream = Stream.of("A", "B", "C", "D");
boolean match = stream.allMatch(s -> s.contains("C"));
System.out.println(match);
//输出
false

10.map转换

map方法可以帮我们做元素转换,比如一个元素所有字母转化为大写,又或者把获取一个元素对象的某个属性,demo如下:

List<String> list = Arrays.asList("jay", "tianluo");
//转化为大写
List<String> upperCaselist = list.stream().map(String::toUpperCase).collect(Collectors.toList());
upperCaselist.forEach(System.out::println);

11.Reduce

Reduce可以合并流的元素,并生成一个值

int sum = Stream.of(1, 2, 3, 4).reduce(0, (a, b) -> a + b);
System.out.println(sum);

12.peek 打印个日志

peek()方法是一个中间Stream操作,有时候我们可以使用peek来打印日志。

List<String> result = Stream.of("程序员", "小男孩", "捡瓶子")
            .filter(a -> a.contains("程序员"))
            .peek(a -> System.out.println("你好:" + a)).collect(Collectors.toList());
System.out.println(result);
//运行结果
你好:程序员
你好:小男孩
[程序员]

13.Max,Min最大最小

使用lambda流求最大,最小值,非常方便。

List<UserInfo> userInfoList = new ArrayList<>();
userInfoList.add(new UserInfo(1L, "小男孩", 18));
userInfoList.add(new UserInfo(3L, "捡瓶子", 26));
userInfoList.add(new UserInfo(2L, "程序员", 27));
Optional<UserInfo> maxAgeUserInfoOpt = userInfoList.stream().max(Comparator.comparing(UserInfo::getAge));
maxAgeUserInfoOpt.ifPresent(userInfo -> System.out.println("max age user:" + userInfo));
Optional<UserInfo> minAgeUserInfoOpt = userInfoList.stream().min(Comparator.comparing(UserInfo::getAge));
minAgeUserInfoOpt.ifPresent(userInfo -> System.out.println("min age user:" + userInfo));
//运行结果
max age user:UserInfo{userId=2, userName='程序员', age=27}
min age user:UserInfo{userId=1, userName='小男孩', age=18}

14.count统计

一般count()表示获取流数据元素总数。

List<UserInfo> userInfoList = new ArrayList<>();
userInfoList.add(new UserInfo(1L, "小男孩", 18));
userInfoList.add(new UserInfo(3L, "捡瓶子", 26));
userInfoList.add(new UserInfo(2L, "程序员", 27));
long count = userInfoList.stream().filter(user -> user.getAge() > 18).count();
System.out.println("大于18岁的用户:" + count);
//输出
大于18岁的用户:2

15. 常用函数式接口

其实lambda离不开函数式接口,我们来看下JDK8常用的几个函数式接口:

Function<T, R>(转换型): 接受一个输入参数,返回一个结果
Consumer<T> (消费型): 接收一个输入参数,并且无返回操作
Predicate<T> (判断型): 接收一个输入参数,并且返回布尔值结果
Supplier<T> (供给型): 无参数,返回结果

Function<T, R> 是一个功能转换型的接口,可以把将一种类型的数据转化为另外一种类型的数据

private void testFunction() {
    //获取每个字符串的长度,并且返回
    Function<String, Integer> function = String::length;
    Stream<String> stream = Stream.of("程序员", "小男孩", "捡瓶子");
    Stream<Integer> resultStream = stream.map(function);
    resultStream.forEach(System.out::println);
}

Consumer是一个消费性接口,通过传入参数,并且无返回的操作

private void testComsumer() {
        //获取每个字符串的长度,并且返回
        Consumer<String> comsumer = System.out::println;
        Stream<String> stream = Stream.of("程序员", "小男孩", "捡瓶子");
        stream.forEach(comsumer);
    }

Predicate是一个判断型接口,并且返回布尔值结果.

private void testPredicate() {
    //获取每个字符串的长度,并且返回
    Predicate<Integer> predicate = a -> a > 18;
    UserInfo userInfo = new UserInfo(2L, "程序员", 27);
    System.out.println(predicate.test(userInfo.getAge()));
}

Supplier是一个供给型接口,无参数,有返回结果。

private void testSupplier() {
    Supplier<Integer> supplier = () -> Integer.valueOf("666");
    System.out.println(supplier.get());
}

这几个函数在日常开发中,也是可以灵活应用的,比如我们DAO操作完数据库,是会有个result的整型结果返回。我们就可以用Supplier来统一判断是否操作成功。如下:

private void saveDb(Supplier<Integer> supplier) {
    if (supplier.get() > 0) {
        System.out.println("插入数据库成功");
    }else{
        System.out.println("插入数据库失败");
    }
}
@Test
public void add() throws Exception {
    Course course=new Course();
    course.setCname("java");
    course.setUserId(100L);
    course.setCstatus("Normal");
    saveDb(() -> courseMapper.insert(course));
}

最后


本文我们介绍了,工作中常用的十几种Lambda 表达式的使用。如果对你有帮助,麻烦给个三连(点赞、在看、转发)。一起加油

相关文章
|
域名解析 网络协议 Java
Java面向对象编程(46)
Java面向对象编程(46)
68 0
|
Java 程序员 数据安全/隐私保护
Java面向对象编程(30)
Java面向对象编程(30)
46 0
|
8月前
|
JavaScript 前端开发 Scala
谈一谈你理解的函数式编程?
谈一谈你理解的函数式编程?
66 0
|
1天前
|
Java API 数据处理
Lambda表达式与函数式工具应用详解
通过继承,C++可以实现代码重用、扩展类的功能并支持多态性。理解继承的类型、重写与重载、多重继承及其相关问题,对于掌握C++面向对象编程至关重要。希望本文能为您的C++学习和开发提供实用的指导。
23 9
|
3月前
|
安全 程序员 编译器
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
97 11
|
8月前
|
Java 程序员 数据安全/隐私保护
Java面向对象编程:类和对象详解
Java面向对象编程:类和对象详解
63 0
|
8月前
|
Java C#
匿名类大揭秘:代码背后的奥秘
匿名类大揭秘:代码背后的奥秘
59 2
|
8月前
|
算法 程序员 编译器
C ++匿名函数:揭开C++ Lambda表达式的神秘面纱
C ++匿名函数:揭开C++ Lambda表达式的神秘面纱
227 0
|
8月前
|
设计模式 Java 程序员
为什么程序员喜欢用Lambda表达式?
“lambda 表达式”是一段可以传递的代码,因此它可以被执行一次或多次。在学习语法(甚至包括一些奇怪的术语)之前,我们先回顾一下之前在Java 中一直使用的相似的代码块。
67 0
|
并行计算 Java 编译器
教你精通Java语法之第十五章、Lambda表达式
Lambda表达式的优点很明显,在代码层次上来说,使代码变得非常的简洁。缺点也很明显,代码不易读。1. 代码简洁,开发迅速2. 方便函数式编程3. 非常容易进行并行计算4. Java 引入 Lambda,改善了集合操作1. 代码可读性变差2. 在非并行计算中,很多计算未必有传统的 for 性能要高3. 不容易进行调试。
64 0