Collectors扩展处理BigDecimal数据

Collectors默认求和,求平均,求最大和最小只有Int/Long/Double三种数据类型,但我们常用价格类型基本都是使用的是 BigDecimal类型,对于这种情况需要扩展Collectors实现。
Collectors扩展处理BigDecimal数据类

import com.alibaba.mos.api.ToBigDecimalFunction;

import java.math.BigDecimal;
import java.math.MathContext;
import java.util.Collections;
import java.util.Set;
import java.util.function.BiConsumer;
import java.util.function.BinaryOperator;
import java.util.function.Function;
import java.util.function.Supplier;
import java.util.stream.Collector;

public class CollectorsUtil {

/**
 *
 */
static final Set<Collector.Characteristics> CH_NOID = Collections.emptySet();

private CollectorsUtil() {
}

/**
 * @param <I>
 * @param <R>
 * @return
 */
@SuppressWarnings("unchecked")
private static <I, R> Function<I, R> castingIdentity() {
    return i -> (R) i;
}

/**
 * Simple implementation class for {@code Collector}.
 *
 * @param <T> the type of elements to be collected
 * @param <R> the type of the result
 */
static class CollectorImpl<T, A, R> implements Collector<T, A, R> {
    private final Supplier<A> supplier;
    private final BiConsumer<A, T> accumulator;
    private final BinaryOperator<A> combiner;
    private final Function<A, R> finisher;
    private final Set<Characteristics> characteristics;

    CollectorImpl(Supplier<A> supplier, BiConsumer<A, T> accumulator, BinaryOperator<A> combiner,
                  Function<A, R> finisher, Set<Characteristics> characteristics) {
        this.supplier = supplier;
        this.accumulator = accumulator;
        this.combiner = combiner;
        this.finisher = finisher;
        this.characteristics = characteristics;
    }

    CollectorImpl(Supplier<A> supplier, BiConsumer<A, T> accumulator, BinaryOperator<A> combiner,
                  Set<Characteristics> characteristics) {
        this(supplier, accumulator, combiner, castingIdentity(), characteristics);
    }

    @Override
    public BiConsumer<A, T> accumulator() {
        return accumulator;
    }

    @Override
    public Supplier<A> supplier() {
        return supplier;
    }

    @Override
    public BinaryOperator<A> combiner() {
        return combiner;
    }

    @Override
    public Function<A, R> finisher() {
        return finisher;
    }

    @Override
    public Set<Characteristics> characteristics() {
        return characteristics;
    }
}

/**
 * 求和
 *
 * @param mapper
 * @param <T>
 * @return
 */
public static <T> Collector<T, ?, BigDecimal> summingBigDecimal(ToBigDecimalFunction<? super T> mapper) {
    return new CollectorImpl<>(
            () -> new BigDecimal[]{new BigDecimal(0)},
            (a, t) -> {
                a[0] = a[0].add(mapper.applyAsBigDecimal(t), MathContext.DECIMAL32);
            },
            (a, b) -> {
                a[0] = a[0].add(b[0], MathContext.DECIMAL32);
                return a;
            },
            a -> a[0], CH_NOID);
}


/**
 * 求最大,这里的最小MIN值,作为初始条件判断值,如果某些数据范围超过百亿以后,可以根据需求换成Long.MIN_VALUE或者Double.MIN_VALUE
 *
 * @param mapper
 * @param <T>
 * @return
 */
public static <T> Collector<T, ?, BigDecimal> maxBy(ToBigDecimalFunction<? super T> mapper) {
    return new CollectorImpl<>(
            () -> new BigDecimal[]{new BigDecimal(Integer.MIN_VALUE)},
            (a, t) -> {
                a[0] = a[0].max(mapper.applyAsBigDecimal(t));
            },
            (a, b) -> {
                a[0] = a[0].max(b[0]);
                return a;
            },
            a -> a[0], CH_NOID);
}

/**
 * 求最小,这里的最大MAX值,作为初始条件判断值,如果某些数据范围超过百亿以后,可以根据需求换成Long.MAX_VALUE或者Double.MAX_VALUE
 *
 * @param mapper
 * @param <T>
 * @return
 */
public static <T> Collector<T, ?, BigDecimal> minBy(ToBigDecimalFunction<? super T> mapper) {
    return new CollectorImpl<>(
            () -> new BigDecimal[]{new BigDecimal(Integer.MAX_VALUE)},
            (a, t) -> {
                a[0] = a[0].min(mapper.applyAsBigDecimal(t));
            },
            (a, b) -> {
                a[0] = a[0].min(b[0]);
                return a;
            },
            a -> a[0], CH_NOID);
}

/**
 * 求平均,并且保留小数
 *
 * @param newScale     保留小数位数
 * @param roundingMode 小数处理方式
 *                     #ROUND_UP 进1
 *                     #ROUND_DOWN 退1
 *                     #ROUND_CEILING  进1截取:正数则ROUND_UP,负数则ROUND_DOWN
 *                     #ROUND_FLOOR  退1截取:正数则ROUND_DOWN,负数则ROUND_UP
 *                     #ROUND_HALF_UP >=0.5进1
 *                     #ROUND_HALF_DOWN >0.5进1
 *                     #ROUND_HALF_EVEN
 *                     #ROUND_UNNECESSARY
 */
public static <T> Collector<T, ?, BigDecimal> averagingBigDecimal(ToBigDecimalFunction<? super T> mapper, int newScale, int roundingMode) {
    return new CollectorImpl<>(
            () -> new BigDecimal[]{new BigDecimal(0), new BigDecimal(0)},
            (a, t) -> {
                a[0] = a[0].add(mapper.applyAsBigDecimal(t));
                a[1] = a[1].add(BigDecimal.ONE);
            },
            (a, b) -> {
                a[0] = a[0].add(b[0]);
                return a;
            },
            a -> a[0].divide(a[1], MathContext.DECIMAL32).setScale(newScale, roundingMode), CH_NOID);
}

}

原文链接:https://blog.csdn.net/zhyzcl/article/details/107306120

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值