JAVA处理类似饼状图占比和100%问题,采用最大余额法

前言:

在做数据统计报表的时候,有两种方式解决占比总和达不到100%或者超过100%问题。
第一种方式是前端echart图自带的算分框架
第二种方式是java后端取处理这个问题。

现存问题:

前端不通过饼状图的方式去展示各个分类的占比累加和为100%问题,
由于各种原因需要后端来计算每个项所占的百分比,但是会发现计算的各项百分比合计不绝对是100,不能简单的对各项使用四舍五入法,舍九法,进一法等

处理方式:

后端采用最大余额法可以解决问题。

最大余额法:

最大余额方法是比例代表制投票制度下,一种议席分配的方法,相对于最高均数方法。

透过最大余额方法,候选人须以名单参选,每份名单的人数最多可达至相关选区内的议席数目。候选人在名单内按优先次序排列。
选民投票给一份名单,而不是个别候选人。投票结束后,把有效选票除以数额。一份名单每取得数额1倍的票数,便能获分配一个议席。
每份名单的候选人按原先订立的顺序当选。

如此类推、将议席分配至每份名单的余额,均比数额为低的时候,则从最大余额者顺序分配余下议席;最大余额方法因而得名。

举例思路:

封装工具类:通用性高,直接应用到项目中:亲测有效

java 复制代码
/**
 * @Description:计算占比通用类
 * 入参封装传入占比数量、分组名即可
 */
@Data
@NoArgsConstructor
public class PercentVo {

    // 占比数量
    private BigDecimal number;

    // 分组名
    private String name;

    // 占比
    private BigDecimal percent;

    // 小数位数值
    private BigDecimal point;

    public PercentVo(BigDecimal number, String name) {
        this.number = number;
        this.name = name;
    }
}

核心工具类

java 复制代码
public class PercentUtils {

    /**
     * 最大余额法-处理百分号占比问题
     * @param total 总数值
     * @param list 各分组数据列表
     * @return
     */
    public static  Map<String, List<PercentVo>> calculatePercent(BigDecimal total,List<PercentVo> list) {

        if(CollectionUtils.isEmpty(list)||Objects.isNull(total))
            throw new BusinessException("计算占比-数据有误");

        if(total.compareTo(BigDecimal.ZERO)==0){
            list.forEach(o->{o.setPercent(BigDecimal.ZERO);o.setPoint(BigDecimal.ZERO);});
            return list.stream().collect(Collectors.groupingBy(PercentVo::getName));
        }

        if(list.size()<2) {
            PercentVo vo = list.get(0);
            if(Objects.isNull(vo))
                throw new BusinessException("计算占比-数据有误");
            BigDecimal percent = vo.getNumber().divide(total,10,RoundingMode.HALF_UP)
                                               .multiply(BigDecimal.valueOf(100))
                                               .setScale(0,RoundingMode.HALF_UP);
            vo.setPercent(percent);
            return list.stream().collect(Collectors.groupingBy(PercentVo::getName));
        }

        for (PercentVo percentVo : list) {
            BigDecimal valueInt = percentVo.getNumber().multiply(BigDecimal.valueOf(100)).divide(total,10, BigDecimal.ROUND_DOWN);
            //设置对应的百分比
            percentVo.setPercent(valueInt.setScale(0,BigDecimal.ROUND_DOWN));
            //获取小数点后的值
            percentVo.setPoint(valueInt.remainder(BigDecimal.valueOf(1)));
        }
        //求和:当前各项百分比合计。由于我们舍弃了小数位,所以该合计只会小于等于100
        BigDecimal reduce = list.stream().map(PercentVo::getPercent).reduce(BigDecimal.ZERO, BigDecimal::add);
        while (reduce.compareTo(BigDecimal.valueOf(100))<0) {
            //找出小数余额最大的组,对其进行加1
            PercentVo max = list.stream().max(Comparator.comparing(PercentVo::getPoint)).get();
            max.setPercent(max.getPercent().add(BigDecimal.ONE));
            //当前这个数已经加1了,不应该参与下一轮的竞选
            max.setPoint(BigDecimal.ZERO);
            reduce = reduce.add(BigDecimal.ONE);
        }
       return list.stream().collect(Collectors.groupingBy(PercentVo::getName));
    }
}
相关推荐
今天背单词了吗98018 分钟前
算法学习笔记:11.冒泡排序——从原理到实战,涵盖 LeetCode 与考研 408 例题
java·学习·算法·排序算法·冒泡排序
Brookty25 分钟前
【操作系统】进程(二)内存管理、通信
java·linux·服务器·网络·学习·java-ee·操作系统
风象南25 分钟前
SpringBoot 与 HTMX:现代 Web 开发的高效组合
java·spring boot·后端
倔强的小石头_3 小时前
【C语言指南】函数指针深度解析
java·c语言·算法
kangkang-7 小时前
PC端基于SpringBoot架构控制无人机(三):系统架构设计
java·架构·无人机
界面开发小八哥9 小时前
「Java EE开发指南」如何用MyEclipse创建一个WEB项目?(三)
java·ide·java-ee·myeclipse
idolyXyz9 小时前
[java: Cleaner]-一文述之
java
一碗谦谦粉9 小时前
Maven 依赖调解的两大原则
java·maven
netyeaxi10 小时前
Java:使用spring-boot + mybatis如何打印SQL日志?
java·spring·mybatis
收破烂的小熊猫~10 小时前
《Java修仙传:从凡胎到码帝》第四章:设计模式破万法
java·开发语言·设计模式