Java實(shí)現(xiàn)按比抽獎(jiǎng)功能
需求是要做幾個(gè)小游戲的抽獎(jiǎng)功能,需要根據(jù)不同的游戲有不同的抽獎(jiǎng)規(guī)則,其中也有很多共性,可歸納為只按獎(jiǎng)品占比抽取、獎(jiǎng)品占比與獎(jiǎng)品數(shù)量抽取、分段抽取,為方便起見將這些的抽獎(jiǎng)的規(guī)則統(tǒng)一封裝到了工具類中。抽獎(jiǎng)的核心邏輯使用的叫做離散算法實(shí)現(xiàn)的。
一.概述
使用離散算法即根據(jù)獎(jiǎng)品占比進(jìn)行分段,然后再產(chǎn)生隨機(jī)數(shù)匹配所對(duì)應(yīng)的區(qū)間。
首先定義Prize獎(jiǎng)品實(shí)體類,類中有prizeName(獎(jiǎng)品名稱)、prizeWeight(獎(jiǎng)品比重)、prizeCount(獎(jiǎng)品數(shù)量)屬性,下面是核心的代碼:
/** * 按比例隨機(jī)抽取一項(xiàng) * @param list 獎(jiǎng)品列表 * @return 類型值 */ public static String ratioExtract(List<Prize> list) { //非空判斷 if (list==null || list.size()<1) { return null; } //占比之和 double sum=0.00; //分段數(shù)組(20,30,60) double[] subArray=new double[list.size()+1]; //將概率分段 for (int i = 0; i < list.size(); i++) { subArray[i]=sum; //這里除要考慮獎(jiǎng)品所占比重外還要將獎(jiǎng)品數(shù)量計(jì)算分段其中 sum+=list.get(i).getPrizeWeight()*list.get(i).getPrizeCount(); } //加上取最大的值 subArray[subArray.length-1]=sum; /* 產(chǎn)生隨機(jī)數(shù) */ Random random=new Random(); double rand = random.nextDouble()*sum; //返回字符 String field=null; for (int i = 0; i < subArray.length; i++) { if (i==subArray.length-1) { return field; } if (rand>=subArray[i] && rand<subArray[i+1]) { field=list.get(i).getPrizeName(); break; } } return field; }
二、測(cè)試
以下是完整的抽獎(jiǎng)工具類
import lombok.Data; import org.apache.commons.lang.math.RandomUtils; import java.util.List; import java.util.Random; /** * @Description: 抽獎(jiǎng)工具類 * @author: xiake * @Date: 2020/1/5 13:23 * @ModifiedDate:2020/1/5 13:23 * @Copyright: miaoxaike.com */ public class PrizeMathRandom { /** * 按比例隨機(jī)抽取一項(xiàng) * @param fieldArray 類型值數(shù)組 * @param proportions 與類型值對(duì)應(yīng) 的占比值 * @return 類型值 */ public static String ratioExtract(String[] fieldArray,double[] proportions) { //判斷兩個(gè)數(shù)組長(zhǎng)度是否相等 if(fieldArray.length!=proportions.length) { return "兩數(shù)組長(zhǎng)度不相等,無法執(zhí)行"; } //占比之和 double sum=0.00; //分段數(shù)組(20,30,60) double[] subArray=new double[proportions.length+1]; //將概率分段 for (int i = 0; i < proportions.length; i++) { subArray[i]=sum; sum+=proportions[i]; } //加上取最大的值 subArray[subArray.length-1]=sum; Random random=new Random(); /* 產(chǎn)生隨機(jī)數(shù) 區(qū)間為 (0,sum)*/ double rand = random.nextDouble()*sum; //返回字符 String field=null; for (int i = 0; i < subArray.length; i++) { if (rand>=subArray[i] && rand<subArray[i+1]) { field=fieldArray[i]; } } return field; } /** * 按比例隨機(jī)抽取一項(xiàng) * @param list 獎(jiǎng)品列表 * @return 類型值 */ public static String ratioExtract(List<Prize> list) { //非空判斷 if (list==null || list.size()<1) { return null; } //占比之和 double sum=0.00; //分段數(shù)組(20,30,60) double[] subArray=new double[list.size()+1]; //將概率分段 for (int i = 0; i < list.size(); i++) { subArray[i]=sum; sum+=list.get(i).getPrizeWeight()*list.get(i).getPrizeCount(); } //加上取最大的值 subArray[subArray.length-1]=sum; /* 產(chǎn)生隨機(jī)數(shù) */ Random random=new Random(); double rand = random.nextDouble()*sum; //返回字符 String field=null; for (int i = 0; i < subArray.length; i++) { if (i==subArray.length-1) { return field; } if (rand>=subArray[i] && rand<subArray[i+1]) { field=list.get(i).getPrizeName(); break; } } return field; } /** * 雙重分段抽取, * @param fieldArray 分段數(shù)組, 參數(shù)值用"-"組裝(例: {"6-14","14-23","23-32","32-40"}) * @param proportions 每段出現(xiàn)的概率 * @return 返回按比例抽取后, 分段范圍內(nèi)的隨機(jī)一個(gè)值 */ public static Integer ratioExtractDouble(String[] fieldArray,double[] proportions) { String string = ratioExtract(fieldArray,proportions); String[] split = string.split("-"); int result = RandomUtils.nextInt(Integer.parseInt(split[1]))+Integer.parseInt(split[0]); return result; } @Data @NoArgsConstructor @AllArgsConstructor class Prize{ //獎(jiǎng)品名稱 private String prizeName; //獎(jiǎng)品占比 private double prizeWeight; //獎(jiǎng)品數(shù)量 private int prizeCount; } }
除了核心的實(shí)現(xiàn)方法外另外還補(bǔ)充了兩個(gè)擴(kuò)充的方法為滿足游戲規(guī)則所用。下面簡(jiǎn)單做個(gè)測(cè)試
public static void main(String[] args) { //初始化獎(jiǎng)品信息 List<Prize> prizeList=new ArrayList<>(); prizeList.add(new Prize("一等獎(jiǎng)",1,1)); prizeList.add(new Prize("二等獎(jiǎng)",3,4)); prizeList.add(new Prize("三等獎(jiǎng)",6,5)); for (int i = 0; i < 12; i++) { Prize prize = ratioExtract(prizeList); if (prize!=null){ System.out.println("第"+(i+1)+"次,抽中 "+prize.getPrizeName()+" 剩余獎(jiǎng)品數(shù)量="+prize.getPrizeCount()); }else { System.out.println("第"+(i+1)+"次,獎(jiǎng)品已抽完"); } } }
運(yùn)行效果如下
實(shí)現(xiàn)的方法很簡(jiǎn)單,可能還有些不合理的地方,但也足以滿足當(dāng)前需求了。基本上都是對(duì)數(shù)組與隨機(jī)數(shù)的使用就不詳細(xì)講解了,有問題歡迎在評(píng)論區(qū)留言!
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- 基于Java實(shí)現(xiàn)抽獎(jiǎng)系統(tǒng)
- 簡(jiǎn)單實(shí)現(xiàn)java抽獎(jiǎng)系統(tǒng)
- 純java代碼實(shí)現(xiàn)抽獎(jiǎng)系統(tǒng)
- Java抽獎(jiǎng)算法第二例
- Java實(shí)現(xiàn)雙色球抽獎(jiǎng)隨機(jī)算法示例
- Java簡(jiǎn)易抽獎(jiǎng)系統(tǒng)小項(xiàng)目
- Java抽獎(jiǎng)?chuàng)屬?gòu)算法
- APP轉(zhuǎn)盤抽獎(jiǎng)Java服務(wù)端接口詳解
- java實(shí)現(xiàn)幸運(yùn)抽獎(jiǎng)系統(tǒng)
- java實(shí)現(xiàn)抽獎(jiǎng)功能解析
相關(guān)文章
如何基于ThreadPoolExecutor創(chuàng)建線程池并操作
這篇文章主要介紹了如何基于ThreadPoolExecutor創(chuàng)建線程池并操作,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-11-11Java基于ServletContextListener實(shí)現(xiàn)UDP監(jiān)聽
這篇文章主要介紹了Java基于ServletContextListener實(shí)現(xiàn)UDP監(jiān)聽,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12詳解基于java的Socket聊天程序——客戶端(附demo)
這篇文章主要介紹了詳解基于java的Socket聊天程序——客戶端(附demo),客戶端設(shè)計(jì)主要分成兩個(gè)部分,分別是socket通訊模塊設(shè)計(jì)和UI相關(guān)設(shè)計(jì)。有興趣的可以了解一下。2016-12-12集合嵌套之ArrayList嵌套ArrayList實(shí)例
下面小編就為大家?guī)硪黄锨短字瓵rrayList嵌套ArrayList實(shí)例。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-08-08springboot項(xiàng)目mapper無法自動(dòng)裝配未找到?UserMapper?類型的Bean解決辦法
這篇文章給大家介紹了springboot項(xiàng)目mapper無法自動(dòng)裝配,未找到?‘userMapper‘?類型的?Bean解決辦法(含報(bào)錯(cuò)原因),文章通過圖文結(jié)合的方式介紹的非常詳細(xì),具有一定的參考價(jià)值,需要的朋友可以參考下2024-02-02JavaWeb項(xiàng)目部署到服務(wù)器詳細(xì)步驟詳解
這篇文章主要介紹了JavaWeb項(xiàng)目如何部署到服務(wù)器,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04springboot中使用過濾器,jsoup過濾XSS腳本詳解
這篇文章主要介紹了springboot中使用過濾器,jsoup過濾XSS腳本詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12