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

前言:

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

现存问题:

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

处理方式:

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

最大余额法:

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

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

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

举例思路:
在这里插入图片描述

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

/*** @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;}
}

核心工具类

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)));}//求和:当前各项百分比合计。由于我们舍弃了小数位,所以该合计只会小于等于100BigDecimal reduce = list.stream().map(PercentVo::getPercent).reduce(BigDecimal.ZERO, BigDecimal::add);while (reduce.compareTo(BigDecimal.valueOf(100))<0) {//找出小数余额最大的组,对其进行加1PercentVo 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));}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/442729.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java学习之基础语法

Java学习之基础语法 本文主要是对于有了其他语言基础的人总结的资料&#xff0c;因此本文只写出了Java与C语言&#xff0c;C等语言的区别之处与部分重点。 1.基础语法&#xff1a; 1.1.包与类&#xff1a; 1.1.1.包&#xff1a; 在Java中&#xff0c;包&#xff08;packag…

【C++】类和对象_1_定义和定义域

1.类的定义&#xff1a; class className {//类体&#xff1a;由成员函数和成员变量组成 };//注意后面的分号class为定义类的关键字&#xff0c;ClassName为类的名字&#xff0c;{}中为类的主体&#xff0c;注意类定义结束时后面分号不能省略。 类体中内容称为类的成员&#x…

docker安全与https协议

一、docker存在的安全问题 1、docker 自身漏洞 docker 应用本身实现上会有代码缺陷&#xff0c;docker 历史版本共有超过 20 项漏洞 2、docker公有仓库安全问题 docker 提供了 docker hub&#xff0c;可以让用户上传创建的镜像&#xff0c;以便其他用户下载&#xff0c;快速…

05 MyBatis之表关系的声明+事务+SqlSession三件套的作用域

MyBatis 支持一对一&#xff0c;一对多&#xff0c;多对多查询。XML 文件和注解都能实现关系的操作。多对多实质就是一对多 1. 表关系的维护 1.1 One一对一 一对一查询和多表(两表)查询很相似, 都能查询两表的全部属性 区别是一对一可以在对象中嵌套对象, 呈现包含关系; 多表…

elementUI之用label传值给v-model的el-radio

如图 这个控件比较特别&#xff0c;是用label作为传给v-model的值&#xff0c;其他控件大多都是用value来传值。 el-radio-group的作用是将其中的单选控件作为一组。

1 计算机网络概述(一):概述

目录 目标1 计算机网络概述1.1 计算机网络的定义和演变1、计算机网络的定义2、计算机网络的演变3、网络发展的里程碑4、我国的网络发展 1.2 计算机网络的分类、组成和网络性能1、计算机网络的分类2、计算机网络的组成3、网络主要性能 目标 了解计算机网络的产生和发展过程了解…

chromedriver安装和环境变量配置

chromedriver 1、安装2、【重点】环境变量配置&#xff08;1&#xff09;包的复制&#xff1a;&#xff08;2&#xff09;系统环境变量配置 3、验证 1、安装 网上随便搜一篇chromedriver的安装文档即可。这里是一个快速链接 特别提醒&#xff1a;截止2024.1.30&#xff0c;chr…

第 7 章:进程间通信

在本章中&#xff0c;您将了解Linux提供的进程间通信&#xff08;IPC&#xff09;接口。通过它们&#xff0c;您将全面了解如何满足系统和软件需求。首先&#xff0c;您将学习消息队列&#xff08;MQs&#xff09;&#xff0c;这是对[第3章]中关于管道的讨论的延续。此外&#…

[ESP32]在Thonny IDE中,如何將MicroPython firmware燒錄到ESP32開發板中?

[ESP32 I MicroPython] Flash Firmware by Thonny(4.1.4) IDE 正常安裝流程&#xff0c;可參考上述影片。然而&#xff0c;本篇文章主要是紀錄安裝過程遇到的bug, 供未來查詢用&#xff0c;也一併供有需要的同好參考。 問題:安裝後&#xff0c;Thonny互動介面顯示一堆亂碼和co…

【C++】STL之空间配置器(了解)

一、什么是空间配置器 空间配置器 &#xff0c;顾名思义就是为各个容器高效的管理空间&#xff08;空间的申请与回收&#xff09;的&#xff0c;在默默地工作。虽然在常规使用 STL 时&#xff0c;可能用不到它&#xff0c;但站在学习研究的角度&#xff0c;学习它的实现原理对…

2024年【陕西省安全员A证】考试资料及陕西省安全员A证考试技巧

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 陕西省安全员A证考试资料考前必练&#xff01;安全生产模拟考试一点通每个月更新陕西省安全员A证考试技巧题目及答案&#xff01;多做几遍&#xff0c;其实通过陕西省安全员A证证考试很简单。 1、【多选题】《陕西省建…

pinctrl/gpio子系统(1)-pinctrl子系统介绍及驱动源码分析

1.简介 在如今的驱动开发工作中&#xff0c;实际上已经很少去对着寄存器手册进行驱动开发了&#xff0c;一般板子拿到手&#xff0c;已经有原厂的驱动开发工程师&#xff0c;在gpio子系统、pinctrl子系统中将自家芯片的引脚适配好了。 我们直接基于设备树已配置好的寄存器值&a…