Java8-无限流补全日期,折线图补零
需求
项目中经常会遇到折线图等随日期、月份变化的趋势图,但有时会出现日期不连续的情况,图表就会有中断,需要在后台将日期数据补全,sql或者程序里都可以处理。
而Java8提供的无限流,可以更方便的实现补全日期的操作,因为日期是连续递增的,看似无限序列。
原始数据
自2020-11-27 过去一周的数据,日期不连续
demo
补全日期数据,没有的日期,数据默认补0
Controller
/**
* 折线图数据
*
* @param preDate 开始日期,不传默认近一周
* @return
*/
@GetMapping("chart")
public List<DailyDataChartVo> getChartData(@RequestParam(value = "date", required = false)
@DateTimeFormat(pattern = "yyyy-MM-dd") LocalDate preDate) {
return this.dailyDataService.getChartData(preDate);
}
Service
/**
* 折线图数据
*
* @param preDate 开始日期,不传默认近一周
* @return
*/
public List<DailyDataChartVo> getChartData(LocalDate preDate) {
if (Objects.isNull(preDate)) {
preDate = LocalDate.now().minusWeeks(1);
}
LocalDate endDate = LocalDate.now();
List<DailyDataChartVo> list = this.dailyDataMapper.selectChartData(preDate, endDate);
return this.completeData(preDate, endDate, list);
}
/**
* 补全数据
*
* @param preDate 开始日期
* @param endDate 截止日期
* @param oldList 未补全的列表
* @return 补全后的列表
*/
private List<DailyDataChartVo> completeData(LocalDate preDate, LocalDate endDate, List<DailyDataChartVo> oldList) {
List<DailyDataChartVo> newList = new ArrayList<>();
if (CollectionUtils.isEmpty(oldList)) {
return newList;
}
//间隔的日期列表
List<LocalDate> dates = this.getRangeDays(preDate, endDate);
Map<LocalDate, DailyDataChartVo> map = oldList.stream()
.collect(Collectors.toMap(DailyDataChartVo::getDate, Function.identity()));
dates.forEach(c -> {
if (map.containsKey(c)) {
newList.add(map.get(c));
} else {
//没有这一天的数据,默认补0
newList.add(new DailyDataChartVo(c, BigDecimal.ZERO));
}
});
return newList;
}
/**
* 获取间隔的日期列表
*
* @param preDate 开始日期
* @param endDate 截止日期
* @return
*/
private List<LocalDate> getRangeDays(LocalDate preDate, LocalDate endDate) {
List<LocalDate> dates = new ArrayList<>();
//间隔的天数
long betweenDays = ChronoUnit.DAYS.between(preDate, endDate);
if (betweenDays < 1) {
//开始日期<=截止日期
return dates;
}
//创建一个从开始日期、每次加一天的无限流,限制到截止日期为止
Stream.iterate(preDate, c -> c.plusDays(1))
.limit(betweenDays + 1)
.forEach(dates::add);
return dates;
}
补全后的数据
[
{
"date": "2020-11-20",
"revenue": 22.88
},
{
"date": "2020-11-21",
"revenue": 93.06
},
{
"date": "2020-11-22",
"revenue": 0
},
{
"date": "2020-11-23",
"revenue": 7.99
},
{
"date": "2020-11-24",
"revenue": 0
},
{
"date": "2020-11-25",
"revenue": 50.98
},
{
"date": "2020-11-26",
"revenue": 0
},
{
"date": "2020-11-27",
"revenue": 0
}
]
核心方法
Stream.iterate():接收一个初始元素seed,生成从seed到f的迭代流
/**
* @param seed 初始元素
* @param f UnaryOperator,函数式接口,接收T类型参数,调用apply后返回T本身,应用于上一个元素以产生新元素
*/
public static<T> Stream<T> iterate(final T seed, final UnaryOperator<T> f) {}
//创建一个从开始日期、每次加一天的无限流,限制到截止日期为止
Stream.iterate(preDate, c -> c.plusDays(1))
.limit(betweenDays + 1)
.forEach(dates::add);
利用无限流,得到连续的日期list,遍历,有该日的数据直接填充,没有默认补0。