java實現(xiàn)微信紅包 拼手氣紅包
本文實例為大家分享了java實現(xiàn)微信紅包的具體代碼,供大家參考,具體內(nèi)容如下
要求
基于BigDecimal類實現(xiàn)微信紅包算法的功能,比如設(shè)置紅包總金額,然后設(shè)置需要生成的紅包個數(shù),為每個紅包隨機(jī)指定金額,最低不能低于0.01元,要求:
1、每個紅包金額隨機(jī)指定
2、每個紅包金額不能低于0.01元
3、要求每個紅包的金額之和恰好等于總金額
4、如果平均每個紅包的金額不足0.01元時拋出一個RedPacketException,提示每個紅包金額不能少于0.01元
實現(xiàn)方法
該題主要考察java常用類中Random、BigDecimal以及ArrayList類綜合使用能力,同時對面向?qū)ο螅ǚ庋b,繼承,多態(tài))技術(shù)進(jìn)行實踐能力考察。
代碼
紅包類
創(chuàng)建一個紅包的基本類型,包含屬性:紅包id,紅包金額; 構(gòu)造器:帶參數(shù),不帶參數(shù); 方法:set方法,get方法;重寫toString;
/** * 紅包類 * @author mrchai * */ public class RedPacket { /**紅包ID*/ private int id; /**紅包金額*/ private BigDecimal money; public RedPacket() { } public RedPacket(int id, BigDecimal money) { super(); this.id = id; this.money = money; } public int getId() { return id; } public void setId(int id) { this.id = id; } public BigDecimal getMoney() { return money; } public void setMoney(BigDecimal money) { this.money = money; } @Override public String toString() { return id+"號用戶獲得"+money+"元"; } }
紅包異常類
創(chuàng)建一個異常類,該異常繼承Exception,設(shè)置一個帶參數(shù)和一個不帶參數(shù)的構(gòu)造器。通過構(gòu)造器直接調(diào)用父類中的方法。
/** * 紅包異常 * @author mrchai */ public class RedpacketException extends Exception{ public RedpacketException() { // TODO Auto-generated constructor stub } public RedpacketException(String msg) { super(msg); } }
紅包管理
實習(xí)題目功能的主要類,genRedPacke()方法分配紅包總金額, randomScale()方法返回一個包含所有紅包金額的比例的數(shù)組。genRedPacke()方法中可以調(diào)用randomScale()方法實現(xiàn)功能。
package com.softeem.lesson18.RedPacket; import java.math.BigDecimal; import java.util.ArrayList; import java.util.Random; public class RedPacketManage { /** 設(shè)置每個紅包最小金額 */ static final BigDecimal MIN = new BigDecimal("0.01"); /** * @double total 總金額 * @int count 紅包個數(shù) * @return 返回生成的所有紅包金額集合 */ public static ArrayList<RedPacket> genRedPacket(double total, int count) throws RedPacketException { // 聲明臨時變量用于存儲所有隨機(jī)的紅包對象 ArrayList<RedPacket> packets = new ArrayList<RedPacket>(); // 計算每個紅包分配最低金額一共需要多少錢 double min = MIN.multiply(new BigDecimal(count)).setScale(2, BigDecimal.ROUND_HALF_EVEN).doubleValue(); if (min > total) { // 紅包金額不夠分配時,拋出異常 throw new RedPacketException("每個紅包金額不能少于0.01元"); } else if (min == total) { // 紅包金額恰好每人只夠分配0.01元,則平均分配 for (int i = 0; i < count; i++) { // 創(chuàng)建紅包對象 RedPacket item = new RedPacket(i + 1, new BigDecimal("0.01")); // 將紅包加入集合 packets.add(item); } } else { // 當(dāng)總金額大于每人最少金額之和時,隨機(jī)分配 // 將總金額包裝為BigDecimal BigDecimal totalMoney = new BigDecimal(total); //先為每人分配最低金額0.01元 //避免因為總金額太少導(dǎo)致有紅包金額為0 for(int i=0;i<count;i++) { packets.add(new RedPacket(i+1, new BigDecimal("0.01"))); } //將總金額設(shè)置為在原來基礎(chǔ)上減去每人最低分配金額 totalMoney = totalMoney.subtract(new BigDecimal(min)); //聲明臨時變量統(tǒng)計當(dāng)前分配的金額總數(shù) BigDecimal now = new BigDecimal(0); // 獲取每個紅包的比例 double[] scale = randomScale(count); // 為前count-1個紅包分配金額 for (int i = 0; i < count - 1; i++) { // 獲取當(dāng)前比例紅包需要分配的金額 BigDecimal item = totalMoney.multiply(new BigDecimal(scale[i])) .setScale(2, BigDecimal.ROUND_HALF_EVEN); //為每人在最低金額基礎(chǔ)上增加隨機(jī)比例金額 packets.get(i).setMoney(packets.get(i).getMoney().add(item)); //累計已分配金額總數(shù) now = now.add(item); } // 剩余的金額給最后一個 //獲取剩余的金額 BigDecimal last = totalMoney.subtract(now); //設(shè)置最后一個紅包的金額為原來基礎(chǔ)上增加剩余的總金額 packets.get(count-1).setMoney(packets.get(count-1).getMoney().add(last).setScale(2, BigDecimal.ROUND_HALF_EVEN)); } return packets; } /** * 隨機(jī)紅包金額比例 * @param count 紅包的份數(shù) * @return 每份紅包的比例數(shù)組 */ private static double[] randomScale(int count) { // 臨時數(shù)組存儲所有紅包的金額比例 double[] scale = new double[count]; Random r = new Random(); double total = 0.0; for (int i = 0; i < count; i++) { // 為每一個元素設(shè)置一個1-100隨機(jī)數(shù) scale[i] = r.nextInt(100) + 1; // 累計所有隨機(jī)的數(shù)值 total += scale[i]; } // 循環(huán)計算每個紅包的金額比例 for (int i = 0; i < count; i++) { scale[i] = scale[i] / total; } return scale; } public static void main(String[] args) throws RedPacketException { ArrayList<RedPacket> list = genRedPacket(0.1, 5); BigDecimal t = new BigDecimal(0); for (RedPacket rp : list) { System.out.println(rp); t= t.add(rp.getMoney()); } System.out.println(t); } }
測試
創(chuàng)建一個元素為RedPacket的ArrayList數(shù)組,遍歷輸出數(shù)組中的對象,因為在紅包類中重寫了toString方法,這里可以直接輸出紅包對象表示紅包獲得者和金額。創(chuàng)建一個BigDecimal對象,累積加上每個紅包的金額,得到總金額。
public static void main(String[] args) throws RedpacketException { ArrayList<RedPacket> list = genRedPacket(0.1, 5); BigDecimal t = new BigDecimal(0); for (RedPacket rp : list) { System.out.println(rp); t= t.add(rp.getMoney()); } System.out.println(t); }
運行結(jié)果
總結(jié)
1.double類型的值可以直接計算,為什么要轉(zhuǎn)換成BigDecimal類型再計算?
答:double類型可以進(jìn)行加減乘除運算,但是會產(chǎn)生一定的誤差,在一些精度要求不高的地方可以直接計算。但是像紅包這種涉及到金額這種對精度要求很高的問題了,顯然無法滿足需要。這時候?qū)⑵滢D(zhuǎn)換成BigDecimal類型,可以有效解決精度的問題。
2.如果自己定義的總金額太低,會不會產(chǎn)生有紅包金額為零的問題?
答:這里代碼的解決思路是:
紅包金額不夠分配時,拋出異常,即總金額小于MIN乘以count時,拋出異常;
紅包金額恰好每人只夠分配0.01元,則平均分配;
當(dāng)總金額大于每人最少金額之和時,隨機(jī)分配:
此時,為了避免因為總金額太少導(dǎo)致有紅包金額為0, 先為每人分配最低金額0.01元,再將總金額設(shè)置為在原來基礎(chǔ)上減去每人最低分配金額。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Linux環(huán)境卸載Centos7自帶的OpenJDK和安裝JDK1.8圖文教程
CentOS系統(tǒng)是開發(fā)者常用的Linux操作系統(tǒng),安裝它時會默認(rèn)安裝自帶的舊版本的OpenJDK,但在開發(fā)者平時開發(fā)Java項目時還是需要完整的JDK,這篇文章主要給大家介紹了關(guān)于Linux環(huán)境卸載Centos7自帶的OpenJDK和安裝JDK1.8的相關(guān)資料,需要的朋友可以參考下2024-07-07解決mybatis使用char類型字段查詢oracle數(shù)據(jù)庫時結(jié)果返回null問題
這篇文章主要介紹了mybatis使用char類型字段查詢oracle數(shù)據(jù)庫時結(jié)果返回null問題的解決方法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價值,需要的朋友可以參考下2018-06-06Java?EasyExcel導(dǎo)出合并單元格的示例詳解
EasyExcel是阿里巴巴開源的一個excel處理框架,以使用簡單、節(jié)省內(nèi)存著稱,這篇文章主要為大家介紹了如何利用EasyExcel導(dǎo)出合并單元格,需要的可以參考下2023-09-09淺談Java中ThreadLocal內(nèi)存泄露的原因及處理方式
內(nèi)存泄漏就是我們申請了內(nèi)存,但是該內(nèi)存一直無法釋放,就會導(dǎo)致內(nèi)存溢出問題,本文詳細(xì)的介紹了ThreadLocal內(nèi)存泄露的原因及處理方式,感興趣的可以了解一下2023-05-05SpringBoot升級3.2報錯Invalid value type for
這篇文章給大家介紹了SpringBoot升級3.2報錯Invalid value type for attribute ‘factoryBeanObjectType‘: java.lang.String的解決方案,文中有詳細(xì)的原因分析,需要的朋友可以參考下2023-12-12