Redian新闻
>
我用Lambda表达式写代码,开发速度提高了10倍!

我用Lambda表达式写代码,开发速度提高了10倍!

公众号新闻

点击上方“芋道源码”,选择“设为星标

管她前浪,还是后浪?

能浪的浪,才是好浪!

每天 10:33 更新文章,每天掉亿点点头发...

源码精品专栏

 
来源:捡田螺的小男孩

前言

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

基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/ruoyi-vue-pro
  • 视频教程:https://doc.iocoder.cn/video/

1. list 转 map

工作中,我们经常遇到listmap的案例。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

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/yudao-cloud
  • 视频教程:https://doc.iocoder.cn/video/

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 分组

提到分组,相信大家都会想起SQLgroup 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 匹配所有元素

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(1234).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<T>是一个消费性接口,通过传入参数,并且无返回的操作

   private void testComsumer() {
        //获取每个字符串的长度,并且返回
        Consumer<String> comsumer = System.out::println;
        Stream<String> stream = Stream.of("程序员芋艿""芋道源码""打代码的芋艿");
        stream.forEach(comsumer);
    }

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

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

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

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

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

    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));
        }


欢迎加入我的知识星球,一起探讨架构,交流源码。加入方式,长按下方二维码噢

已在知识星球更新源码解析如下:

最近更新《芋道 SpringBoot 2.X 入门》系列,已经 101 余篇,覆盖了 MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。

提供近 3W 行代码的 SpringBoot 示例,以及超 4W 行代码的电商微服务项目。

获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。

文章有帮助的话,在看,转发吧。

谢谢支持哟 (*^__^*)

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
这回写代码,不怕有 Bug 了《代码英雄》第五季(2):写代码的地方 | Linux 中国【庭院种菜】蘑菇土种菜的潜在危害当我手写代码,别人却觉得我在画小鸟...3行代码建模,训练速度提升200%?这款时序开源神器PaddleTS太强了!研究了代码质量后,开发速度提高了 2 倍,bug 减少了 15 倍训练速度提高最多5.4倍,谷歌提出RL训练新范式ActorQSwitch成为最易吃灰的电子产品;马斯克要员工默写代码;联发科:坚守价格底线​能让天津人放下煎饼馃子的,也就那碗捞(láo)面了!马斯克“下狠手”裁员推特50%,要求工程师默写代码,员工每周工作84小时让程序员动嘴写代码,Copilot测试新功能「嘿,GitHub!」华为云发布冷启动加速解决方案:助力Serverless计算速度提升90%+万水千山帝王蝶因开发人员误公开源代码,丰田或泄露近30万客户信息好强的谷歌“Excel”,不用自己写代码就能爬虫,网友立马cue微软:慌不慌?华为开发者贡献 Linux 内核补丁,将核心内核函数速度提升 715 倍用1个月重构了同事写的烂代码,我总结出了15条重写烂代码的经验!不写代码,就能快速构建精准的机器学习模型用了这些 IDEA 插件后,我写代码快了 N 倍!公司花50k挖了一个BAT架构师,写代码上线直接内存溢出,被开除了!工农红军每月有10几塊大洋吗100亿参数的语言模型跑不动?MIT华人博士提出SmoothQuant量化,内存需求直降一半,速度提升1.56倍!当 Rust 成为“巨坑”:拖慢开发速度、员工被折磨数月信心全无,无奈还得硬着头皮继续胡渊鸣:import 一个“太极”库,让 Python 代码提速100倍!好消息,Centrelink津贴即将再度提高,数百万澳人受益!这些福利都将提高杭州妈妈晒“天价”电费,比上个月贵了10倍!网友支招:知道这些,一年省下千把块钱松耦合式的权限控制设计,自定义权限表达式马狼,兔狼,黑狼马斯克惊呼吓人的网红机器人,能写论文/编小说/写代码,出毁灭人类计划书[重磅]新变种XBB杀疯!加拿大新冠疫情又反弹!住院率竟是去年同期两倍!新病例少报多达100倍!平均查找性能提高了 715 倍,Linux 6.2 合并 华为 贡献的代码仿照电影情节写代码,程序员偷盗自家公司30万美元后入狱马斯克“下狠手”整顿推特,要求工程师默写代码,员工每周工作84小时畅游法国(23)-王国南大门重磅|新变种XBB杀疯!加拿大新冠疫情又反弹!住院率竟是去年同期两倍!新病例少报多达100倍!
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。