日常開(kāi)發(fā)中,我們很多時(shí)候需要用到Java 8
的Lambda
表達(dá)式,它允許把函數(shù)作為一個(gè)方法的參數(shù),讓我們的代碼更優(yōu)雅、更簡(jiǎn)潔。所以整理了一波工作中,我常用的,有哪些Lambda
表達(dá)式??赐暌欢〞?huì)有幫助的。
工作中,我們經(jīng)常遇到list
轉(zhuǎn)map
的案例。Collectors.toMap
就可以把一個(gè)list
數(shù)組轉(zhuǎn)成一個(gè)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 轉(zhuǎn) map
* 使用Collectors.toMap的時(shí)候,如果有可以重復(fù)會(huì)報(bào)錯(cuò),所以需要加(k1, k2) -> k1
* (k1, k2) -> k1 表示,如果有重復(fù)的key,則保留第一個(gè),舍棄第二個(gè)
*/
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()));
}
}
//運(yùn)行結(jié)果
撿田螺的小男孩
程序員田螺
類似的,還有Collectors.toList()
、Collectors.toSet()
,表示把對(duì)應(yīng)的流轉(zhuǎn)化為list
或者Set
。
從數(shù)組集合中,過(guò)濾掉不符合條件的元素,留下符合條件的元素。
List<UserInfo> userInfoList = new ArrayList<>();
userInfoList.add(new UserInfo(1L, '撿田螺的小男孩', 18));
userInfoList.add(new UserInfo(2L, '程序員田螺', 27));
userInfoList.add(new UserInfo(3L, '撿瓶子的小男孩', 26));
/**
* filter 過(guò)濾,留下超過(guò)18歲的用戶
*/
List<UserInfo> userInfoResultList = userInfoList.stream().filter(user -> user.getAge() > 18).collect(Collectors.toList());
userInfoResultList.forEach(a -> System.out.println(a.getUserName()));
//運(yùn)行結(jié)果
程序員田螺
撿瓶子的小男孩
foreach 遍歷list,遍歷map,真的很絲滑。
/**
* forEach 遍歷集合List列表
*/
List<String> userNameList = Arrays.asList('撿田螺的小男孩', '程序員田螺', '撿瓶子的小男孩');
userNameList.forEach(System.out::println);
HashMap<String, String> hashMap = new HashMap<>();
hashMap.put('公眾號(hào)', '撿田螺的小男孩');
hashMap.put('職業(yè)', '程序員田螺');
hashMap.put('昵稱', '撿瓶子的小男孩');
/**
* forEach 遍歷集合Map
*/
hashMap.forEach((k, v) -> System.out.println(k + ':\t' + v));
//運(yùn)行結(jié)果
撿田螺的小男孩
程序員田螺
撿瓶子的小男孩
職業(yè): 程序員田螺
公眾號(hào): 撿田螺的小男孩
昵稱: 撿瓶子的小男孩
提到分組,相信大家都會(huì)想起SQL
的group by
。我們經(jīng)常需要一個(gè)List做分組操作。比如,按城市分組用戶。在Java8之前,是這么實(shí)現(xiàn)的:
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
分組器,清爽無(wú)比:
Map<String, List<UserInfo>> result = originUserInfoList.stream()
.collect(Collectors.groupingBy(UserInfo::getCity));
工作中,排序的需求比較多,使用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('開(kāi)始降序排序');
/**
* 如果想降序排序,則可以使用加reversed()
*/
userInfoList = userInfoList.stream().sorted(Comparator.comparing(UserInfo::getAge).reversed()).collect(Collectors.toList());
userInfoList.forEach(a -> System.out.println(a.toString()));
//運(yùn)行結(jié)果
UserInfo{userId=1, userName='撿田螺的小男孩', age=18}
UserInfo{userId=3, userName='撿瓶子的小男孩', age=26}
UserInfo{userId=2, userName='程序員田螺', age=27}
開(kāi)始降序排序
UserInfo{userId=2, userName='程序員田螺', age=27}
UserInfo{userId=3, userName='撿瓶子的小男孩', age=26}
UserInfo{userId=1, userName='撿田螺的小男孩', age=18}
distinct
可以去除重復(fù)的元素:
List<String> list = Arrays.asList('A', 'B', 'F', 'A', 'C');
List<String> temp = list.stream().distinct().collect(Collectors.toList());
temp.forEach(System.out::println);
findFirst
很多業(yè)務(wù)場(chǎng)景,我們只需要返回集合的第一個(gè)元素即可:
List<String> list = Arrays.asList('A', 'B', 'F', 'A', 'C');
list.stream().findFirst().ifPresent(System.out::println);
anyMatch
檢查流是否包含至少一個(gè)滿足給定謂詞的元素。
Stream<String> stream = Stream.of('A', 'B', 'C', 'D');
boolean match = stream.anyMatch(s -> s.contains('C'));
System.out.println(match);
//輸出
true
allMatch
檢查流是否所有都滿足給定謂詞的元素。
Stream<String> stream = Stream.of('A', 'B', 'C', 'D');
boolean match = stream.allMatch(s -> s.contains('C'));
System.out.println(match);
//輸出
false
map
方法可以幫我們做元素轉(zhuǎn)換,比如一個(gè)元素所有字母轉(zhuǎn)化為大寫,又或者把獲取一個(gè)元素對(duì)象的某個(gè)屬性,demo
如下:
List<String> list = Arrays.asList('jay', 'tianluo');
//轉(zhuǎn)化為大寫
List<String> upperCaselist = list.stream().map(String::toUpperCase).collect(Collectors.toList());
upperCaselist.forEach(System.out::println);
Reduce可以合并流的元素,并生成一個(gè)值
int sum = Stream.of(1, 2, 3, 4).reduce(0, (a, b) -> a + b);
System.out.println(sum);
peek()
方法是一個(gè)中間Stream
操作,有時(shí)候我們可以使用peek
來(lái)打印日志。
List<String> result = Stream.of('程序員田螺', '撿田螺的小男孩', '撿瓶子的小男孩')
.filter(a -> a.contains('田螺'))
.peek(a -> System.out.println('公眾號(hào):' + a)).collect(Collectors.toList());
System.out.println(result);
//運(yùn)行結(jié)果
公眾號(hào):程序員田螺
公眾號(hào):撿田螺的小男孩
[程序員田螺, 撿田螺的小男孩]
使用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));
//運(yùn)行結(jié)果
max age user:UserInfo{userId=2, userName='程序員田螺', age=27}
min age user:UserInfo{userId=1, userName='撿田螺的小男孩', age=18}
一般count()
表示獲取流數(shù)據(jù)元素總數(shù)。
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
其實(shí)lambda離不開(kāi)函數(shù)式接口,我們來(lái)看下JDK8常用的幾個(gè)函數(shù)式接口:
Function<T, R>
(轉(zhuǎn)換型): 接受一個(gè)輸入?yún)?shù),返回一個(gè)結(jié)果Consumer<T>
(消費(fèi)型): 接收一個(gè)輸入?yún)?shù),并且無(wú)返回操作Predicate<T>
(判斷型): 接收一個(gè)輸入?yún)?shù),并且返回布爾值結(jié)果Supplier<T>
(供給型): 無(wú)參數(shù),返回結(jié)果Function<T, R>
是一個(gè)功能轉(zhuǎn)換型的接口,可以把將一種類型的數(shù)據(jù)轉(zhuǎn)化為另外一種類型的數(shù)據(jù)
private void testFunction() {
//獲取每個(gè)字符串的長(zhǎng)度,并且返回
Function<String, Integer> function = String::length;
Stream<String> stream = Stream.of('程序員田螺', '撿田螺的小男孩', '撿瓶子的小男孩');
Stream<Integer> resultStream = stream.map(function);
resultStream.forEach(System.out::println);
}
Consumer<T>
是一個(gè)消費(fèi)性接口,通過(guò)傳入?yún)?shù),并且無(wú)返回的操作
private void testComsumer() {
//獲取每個(gè)字符串的長(zhǎng)度,并且返回
Consumer<String> comsumer = System.out::println;
Stream<String> stream = Stream.of('程序員田螺', '撿田螺的小男孩', '撿瓶子的小男孩');
stream.forEach(comsumer);
}
Predicate<T>
是一個(gè)判斷型接口,并且返回布爾值結(jié)果.
private void testPredicate() {
//獲取每個(gè)字符串的長(zhǎng)度,并且返回
Predicate<Integer> predicate = a -> a > 18;
UserInfo userInfo = new UserInfo(2L, '程序員田螺', 27);
System.out.println(predicate.test(userInfo.getAge()));
}
Supplier<T>
是一個(gè)供給型接口,無(wú)參數(shù),有返回結(jié)果。
private void testSupplier() {
Supplier<Integer> supplier = () -> Integer.valueOf('666');
System.out.println(supplier.get());
}
這幾個(gè)函數(shù)在日常開(kāi)發(fā)中,也是可以靈活應(yīng)用的,比如我們DAO操作完數(shù)據(jù)庫(kù),是會(huì)有個(gè)result的整型結(jié)果返回。我們就可以用Supplier<T>
來(lái)統(tǒng)一判斷是否操作成功。如下:
private void saveDb(Supplier<Integer> supplier) {
if (supplier.get() > 0) {
System.out.println('插入數(shù)據(jù)庫(kù)成功');
}else{
System.out.println('插入數(shù)據(jù)庫(kù)失敗');
}
}
@Test
public void add() throws Exception {
Course course=new Course();
course.setCname('java');
course.setUserId(100L);
course.setCstatus('Normal');
saveDb(() -> courseMapper.insert(course));
}
本文我們介紹了,工作中常用的十幾種Lambda 表達(dá)式的使用。
聯(lián)系客服