Java8函数式编程实践精华

绪论

从java8开始,我们就可以通过java8中的StrameAPI与Lambda表达式实现函数式编程,可以让代码变得更加高效简洁。现在很多企业的生产代码已经开始使用java8了,对于还没有使用过java8进行的编程的朋友们可以好好的学习一下,我在企业中写java8也有一段时间了,我想把我在实际开发中用到的一些场景与大家分享一下,大部分例子都来源于我开发项目中的实例。其中我开发中用得比较多的是,Optional,SteameAPI与lambda。本文是实战帖,不做过多理论分析,直接讲场景与应用。Java9已经出来一段时间了,我发现我周围的很多做开发的朋友对java8中的一些新特性仍然不是很了解,也没在生产代码中使用过java8,不过我相信,很快,大部分公司都会用上java8的,我觉得java8与java5一样,是jdk历史上重要的一个版本更新,能给java编程带来巨大变化的一个版本。

场景

1.数据类型转换

直接上代码

//拿到所有的processNumber
List<String> processNumbers = param.getListProcessNumbers().stream().
map(QueryStateData::getProcessNumber).collect(Collectors.toList());

上面的代码的作用中把List<Object>,把其中我们需要的一个string字段变成List<String>,map是StringAPI中用来做数据类型转换的,QueryStateData::getProcessNumber是方法引用,作用与object->object.getProcessNumber();一样,collect是收集器,通过收集器,把流转换成我们想要的List<String>

2.对List数据进行数据筛选

//获取paymentChange
Optional<GaPaymentChange> changeOptional = listPaymentChange.stream().filter(p -> p.getTransferNo().equals(std.getProcessNumber())).findFirst();
if(changeOptional.isPresent()){
	GaPaymentChange paymentChange = changeOptional.get();
}

filter从名字上看,就知道是用来过数据过滤的,filter的函数式接口中需返回一个boolean值。findFirst是取其中的第一个,可以返回一个Optional对象。通过get方法我们很轻松的拿到了我们想要的结果。

3.Optional对象的应用

java8中Optional的出现是为了减少空指针异常的。看代码应用:

BigDecimal bigAbmout = Optional.ofNullable(paymentChange.getPaymentChangeMoney()).orElseGet(() -> new BigDecimal("0")).setScale(2, BigDecimal.ROUND_HALF_UP);

当ofNullable值为null时,我们可以给bigAbmout赋一个默认值。orElseGet的传参数是一个函数式接口,上面代码可以把它写成
lambda要返回一个对象。

Optional.ofNullable(paymentChange.getPaymentChangeMoney()).orElse(new BigDecimal("0")).setScale(2, BigDecimal.ROUND_HALF_UP);

orElse与orElseGet不同的是,它的参数直接是目标对象。

4.StreamAPI遍历对象

page.getRows().stream().forEach(x -> x.setOrderTime((DateUtils.dateFormat(x.getCreateTime(),DateUtils.DAT_SHORT_FORMATSS))));

上面的代码就是简单的把集合对象中orderTime的值做一下转换。

5.快速找到集合中想要匹配数据

//是否存在增补单
boolean match = gaPaymentChangeDetailList.stream().anyMatch(detail -> GaConstants.SUPPLEMENT_BILL.equals(detail.getIsSupplementBill()));
            

anyMatch配合lambda表达式可以在集合中快速找到我们想匹配的数据。

6.List中对象转换

List<Menu>menus= menuService.queryMenuList(query);List<MenuTreeVO> menuTrees = menus.stream().map(MenuTreeVO::new).collect(Collectors.toList());

利用StreamAPI两行代码就把List<Menu>转成了 List<MenuTreeVO>,其中的map(MenuTreeVO::new)等价于map(menu->new MenuTreeVO(menu)),所以,实现的一个另一个细节,我重写了MenuTreeVO的构造方法:

public MenuTreeVO(Menu menu) {
	this.id = menu.getId();
	this.pId = menu.getParentId();
	this.name = menu.getName();
	this.open = true;
}

7.枚举中的应用

枚举中我们通常会与一个静态方法通过code去获取它对应的枚举,没有用java8我们一般是通过for循环实现的,看看java8代码:

/**
 * 获取枚举
 * 
 * @param code
 * @return
 */
public static OrderStateEnum getEnum(Integer code) {
	return Stream.of(OrderStateEnum.values()).filter(e ->e.code.equals(code)).findFirst().orElse(null);
}

8.lambda实现函数式接口

我们原来代码中的遇到的函数式接口,我们就可以用lambda来代替,简单的来说,就是new接口的地方。比如MQ发消息,我们传统的做法:

jmsTemplate.send(new MessageCreator() {
	@Override
	public Message createMessage(Session session) throws JMSException {
		return session.createTextMessage(String.valueOf(id));
	}
});

这个函数式接口可以用lambda一行代码轻松实现:

jmsTemplate.send(session -> session.createTextMessage(String.valueOf(id)));

8.去重、求和、最大值、最小值

用法都比较类似,这里就不一一列举了。下面看一个去重的。stream下面的方法也就那么多个,要用的时候可以一个个点来看看,从英文意思我们可大概知道那些方法基本上用来做什么的了。其实很多用法我也是在实际开发中慢慢研究出来的,也没看很多的资源。

List<Color> colors = skusResult.getValue().stream().map(Sku::getColor).distinct().collect(Collectors.toList());

上面是对List<Color>中的Color对象去重,记得重写hashCode()与equeals()方法。

9.对List统计出现次数,按出现次数排序

这个用法也是实际生产中用到的,直接上代码。

//此处是java9之后的快速创建list的写法
var names = List.of("aaa", "aaa", "aaa", "bbb", "bbb", "bbb", "bbb", "ccc");
//按出现次数分组
var sorted = names.stream()
				.collect(Collectors.groupingBy(p -> p, Collectors.counting())).entrySet();

如果我们需要对结果按次数排序,可以如下操作:

var sortedAndGrouping = names.stream().collect(Collectors.groupingBy(p -> p, Collectors.counting())).entrySet().stream()
                .sorted((value1, value2) -> value2.getValue().compareTo(value1.getValue())).map(Map.Entry::getKey).collect(Collectors.toList());

这里实现了分组与排序,并且把list中出现次数最高的放在也第一个元素。

10.List按长度排序

resList.stream().sorted(Comparator.comparingInt(String::length))
                .collect(Collectors.toList());

结语

在实际的开发中,能应用的场景还有很多,我上面列举的一些可能用得会相对多一些,也有很多应用场景我没有提到,java8函数式编程的也许远远不止这些,以后遇到好的场景我再补充,如果大家有好的应用,欢迎留言讨论。

  • 5
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值