在Java8及更高版本中,Lambda表达式的引入极大地提升了编程的简洁性和效率。本文将围绕十个关键场景,展示Lambda如何助力提升开发效率,让代码更加精炼且易于理解。
传统的for-each循环对集合进行遍历虽然直观,但在处理大量数据时显得冗长。例如:
List<String> list = Arrays.asList("a", "b", "c");for (String s : list) { System.out.println(s);}
使用Lambda表达式后,代码变得更加紧凑:
list.forEach(System.out::println);
在以前我们对集合中的元素进行排序时,需要实现Comparable接口,或者使用Comparator比较器,在其中定义排序规则。
Collections.sort(list, new Comparator<String>() { @Override public int compare(String s1, String s2) { return s1.length() - s2.length(); }});
使用Lambda可以进行简化:
List<String> sortedList = list.sort(Comparator.comparingInt(String::length));// 或者Collections.sort(list, (s1, s2) -> s1.length() - s2.length());// 或者Collections.sort(list, Comparator.comparingInt(String::length));
以往的过滤操作以往需要编写繁琐的条件判断。
List<String> filterList = new ArrayList<>();for (String s : list){ if (s.length() >= 4){ filterList.add(s); }}
使用Lambda可以进行简化:
List<String> filterList = list.stream().filter(e -> e.length() >= 4).collect(Collectors.toList());
关于Stream的使用方法请参考:提高Java开发生产力,我选Stream,真香啊
如以下操作,将一个集合变成另外一个集合
List<String> upperCaseList = new ArrayList<>();for (String str : words) { upperCaseList.add(str.toUpperCase());}
而Lambda表达式可用于将集合中的元素直接转换成新的形式:
List<String> upperList = list.stream().map(e -> e.toUpperCase()).collect(Collectors.toList()); upperList = list.stream().map(String::toUpperCase).collect(Collectors.toList()); List<Integer> lengthList = list.stream().map(e -> e.length()).collect(Collectors.toList()); lengthList = list.stream().map(String::length).collect(Collectors.toList());
规约操作,即对一个集合中的元素进行求和,求平均数等
int sum = 0;for (int num : numbers) { sum += num;}
使用Lambda简化
int sum = numbers.stream().mapToInt(Integer::intValue).sum();int sum = numbers.stream().reduce(0, (n1, n2) -> n1 + n2);int sum = numbers.stream().reduce(0, Integr::sum);List<Person> peoples = new ArrayList<>();int ages = peoples.stream().mapToInt(Person::getAge).sum();
关于Stream的使用方法请参考:提高Java开发生产力,我选Stream,真香啊
对一个集合基于特定规则对集合进行分组,即将List<Object>转换为Map<Object, List<Object>>
List<Person> personList = new ArrayList<>();Map<String, List<Person>> groupMap = new HashMap<>();for (Person person : personList) { Integer age = person.getAge(); if (!groupMap.containsKey(age)) { groupMap.put(age, new ArrayList<>()); } groupMap.get(age).add(person);}
使用Lambda简化:
Map<String, List<Person>> groupMap = words.stream() .collect(Collectors.groupingBy(Person::age));
还有另外一种List<Object>转换为Map<Object, Object>:
List<Person> personList = new ArrayList<>();Map<Long, Person> personMap = new HashMap<>();for (Person person : personList) { personMap.put(person.getId(), person);}
使用Lambda简化:
Map<String, Person> groupMap = words.stream() .collect(Collectors.toMap(Person::id, Function.identity(), (e1, e2) -> e1));
关于Stream的使用方法请参考:提高Java开发生产力,我选Stream,真香啊
现在有一个函数式接口:
@FunctionalInterfaceinterface MyInterface{ void doSomething(String s);}
常规做法在使用函数式接口时:
MyInterface myInterface = new MyInterface() { @Override public void doSomething(String s) { System.out.println(s); }};myInterface.doSomething("I am 码农Academy");
使用Lamba进行优化:
MyInterface myInterface = s -> System.out.println(s);myInterface.doSomething("I am 码农Academy");
以往创建线程的方式:
Thread thread = new Thread(new Runnable() { @Override public void run() { System.out.println("Hello, 码农Academy!"); }});
使用Lambda简化后:
Thread thread = new Thread(() -> System.out.println("Hello, 码农Academy!"));// 或者使用线程池方式ExecutorService executor = Executors.newFixedThreadPool(5); executor.execute(() -> longRunningTask());
Optional可以避免空指针异常。
Optional<String> optional = ...;if (optional.isPresent()) { String value = optional.get(); // 处理value}
使用Lambda简化:
Optional<String> optional = ...;optional.ifPresent(value -> handleValue(value));
关于使用Optional解决空指针的用法,可以参考:聊一聊日常开发中如何避免那无处不在的让人头疼的NullPointerException
在处理业务时,我们需要对一个集合进行一系列的操作时,比如如下:`
List<Integer> result = new ArrayList<>();for (String str : list) { if (str.matches("//d+")) { result.add(Integer.parseInt(str)); }}
利用Stream API与Lambda结合,实现链式操作,使代码更清晰易读:
List<Integer> result = list.stream() .filter(str -> str.matches("//d+")) .map(Integer::parseInt) .collect(Collectors.toList());
比如我们使用Lambda结合Stream实现一个去重操作:
/** * 根据学生姓名查询除重复元素 * @param students */ private static void repeatStudentsTest(List<Student> students){ // list 对应的 Stream List<String> repeatStudents = students.stream() // 获得元素出现频率的 Map,键为元素,值为元素出现的次数 .collect(Collectors.toMap(e -> e.getName(), e -> 1, Integer::sum)) // 所有 entry 对应的 Stream .entrySet().stream() // 过滤出元素出现次数大于 1 的 entry(过滤出来的是重复的,若这里条件是等于,即可达到去重的目的) .filter(entry -> entry.getValue()>1) // 获得 entry 的键(重复元素)对应的 Stream .map(entry -> entry.getKey()) // 转化为 List .collect(Collectors.toList()); repeatStudents.forEach(repeatStudent -> { System.out.println(repeatStudent); }); }
关于Stream的使用方法请参考:提高Java开发生产力,我选Stream,真香啊
关于使用Idea开发式,以前对代码断点时确实无法进入到lamda表达式里面,但是随着Idea的升级,已经解决了这个问题,可以在Lambda表达式的内部进行断点
图片
图片
有人可能会认为Lambda表达式的代码阅读起来有些吃力,当然也是可以理解,其主要原因有如下几个方面:
但是,随着Java 8以来函数式编程特性的普及,越来越多的Coder们开始接受并熟练使用Lambda表达式。适当的代码组织、注释和遵循良好的编程规范有助于降低Lambda表达式带来的阅读障碍。并且随着经验的增长和技术背景的丰富,我们会逐渐认识到Lambda表达式的优点,即它可以增强代码的可读性和简洁性,尤其在处理数据流和进行函数组合时。
熟练运用Lambda表达式能够显著提升代码质量与开发效率,使得代码逻辑更加简明扼要,同时也增强了程序的可读性与维护性。不断学习和实践这些技巧,你的开发效率必将迎来质的飞跃。并且Lambda与Stream一起使用才能发挥他们最大的优点。
本文链接:http://www.28at.com/showinfo-26-80890-0.html提高生产力!这10个Lambda表达式必须掌握,开发效率嘎嘎上升!
声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com
上一篇: 分享七个你可能不知道的 Next.js 14 小技巧
下一篇: Exclude 工具类型八个使用技巧