JAVA使用隨機(jī)數(shù)實(shí)現(xiàn)概率抽獎(jiǎng)
本文實(shí)例為大家分享了JAVA使用隨機(jī)數(shù)實(shí)現(xiàn)概率抽獎(jiǎng)的具體代碼,供大家參考,具體內(nèi)容如下
需求
網(wǎng)站現(xiàn)有一抽獎(jiǎng)功能,已經(jīng)定義好獎(jiǎng)品,每個(gè)獎(jiǎng)品都有對(duì)應(yīng)的中獎(jiǎng)概率。通過(guò)獎(jiǎng)品概率隨機(jī)進(jìn)行抽獎(jiǎng)
實(shí)現(xiàn)思路
1、每個(gè)獎(jiǎng)品都有對(duì)應(yīng)的中獎(jiǎng)概率,先對(duì)所有獎(jiǎng)品中獎(jiǎng)概率求和
2、計(jì)算出每個(gè)獎(jiǎng)品在0-1之間所占的區(qū)間塊
3、隨機(jī)產(chǎn)生0-1之間的隨機(jī)數(shù),隨機(jī)數(shù)落在哪個(gè)區(qū)間,就是中獎(jiǎng)哪個(gè)
例如現(xiàn)有以下獎(jiǎng)品:
獎(jiǎng)品A 中獎(jiǎng)概率為 0.1
獎(jiǎng)品B 中獎(jiǎng)概率為 0.01
獎(jiǎng)品C 中獎(jiǎng)概率為 0.001
獎(jiǎng)品D 中獎(jiǎng)概率為 0.8
第一步:求出概率總和 0.1+0.01+0.001+0.8 = 0.911
第二步:計(jì)算每個(gè)獎(jiǎng)品的所占區(qū)間塊
獎(jiǎng)品A: 0.1 / 0.911 = 0.1098
獎(jiǎng)品B: (0.1+0.01)/ 0.911 = 0.1207
獎(jiǎng)品C:(0.1+0.11+0.001)/ 0.911 = 0.1218
獎(jiǎng)品D:(0.1+0.11+0.001+0.8)/ 0.911 = 1
則:
獎(jiǎng)品A的所占區(qū)間為:0~0.1098
獎(jiǎng)品B的所占區(qū)間為:0.1098~0.1207
獎(jiǎng)品C的所占區(qū)間為:0.1207~0.1218
獎(jiǎng)品D的所占區(qū)間為:0.1218~1
代碼如下
/** * 獎(jiǎng)品實(shí)體類 */ public class Award{ public Award(){} public Award(String awardTitle,double probability){ this.awardTitle = awardTitle; this.probability = probability; } /**獎(jiǎng)品ID**/ private String awardId; /**獎(jiǎng)品名**/ private String awardTitle; /**中獎(jiǎng)概率**/ private double probability; public String getAwardId() { return awardId; } public void setAwardId(String awardId) { this.awardId = awardId; } public String getAwardTitle() { return awardTitle; } public void setAwardTitle(String awardTitle) { this.awardTitle = awardTitle; } public double getProbability() { return probability; } public void setProbability(double probability) { this.probability = probability; } }
public class LotteryUtil { /** * 抽獎(jiǎng),獲取中獎(jiǎng)獎(jiǎng)品 * @param awardList 獎(jiǎng)品及中獎(jiǎng)概率列表 * @return 中獎(jiǎng)商品 */ public static Award lottery(List<Award> awardList) { if(awardList.isEmpty()){ throw new AwardListIsEmptyException(); } //獎(jiǎng)品總數(shù) int size = awardList.size(); //計(jì)算總概率 double sumProbability = 0d; for (Award award : awardList) { sumProbability += award.getProbability(); } //計(jì)算每個(gè)獎(jiǎng)品的概率區(qū)間 //例如獎(jiǎng)品A概率區(qū)間0-0.1 獎(jiǎng)品B概率區(qū)間 0.1-0.5 獎(jiǎng)品C概率區(qū)間0.5-1 //每個(gè)獎(jiǎng)品的中獎(jiǎng)率越大,所占的概率區(qū)間就越大 List<Double> sortAwardProbabilityList = new ArrayList<Double>(size); Double tempSumProbability = 0d; for (Award award : awardList) { tempSumProbability += award.getProbability(); sortAwardProbabilityList.add(tempSumProbability / sumProbability); } //產(chǎn)生0-1之間的隨機(jī)數(shù) //隨機(jī)數(shù)在哪個(gè)概率區(qū)間內(nèi),則是哪個(gè)獎(jiǎng)品 double randomDouble = Math.random(); //加入到概率區(qū)間中,排序后,返回的下標(biāo)則是awardList中中獎(jiǎng)的下標(biāo) sortAwardProbabilityList.add(randomDouble); Collections.sort(sortAwardProbabilityList); int lotteryIndex = sortAwardProbabilityList.indexOf(randomDouble); return awardList.get(lotteryIndex); } public static void main(String[] args) { List<Award> awardList = new ArrayList<Award>(); awardList.add(new Award("10個(gè)積分",0.35d)); awardList.add(new Award("33個(gè)積分",0.25d)); awardList.add(new Award("5元紅包",0.002d)); awardList.add(new Award("20元話費(fèi)",0.003d)); awardList.add(new Award("京東100元購(gòu)物卡",0.0005d)); awardList.add(new Award("未中獎(jiǎng)",0.1d)); Map<String,Integer> result = new HashMap<String,Integer>(); for(int i=0;i<10000;i++){ Award award = lottery(awardList); String title = award.getAwardTitle(); Integer count = result.get(title); result.put(title, count == null ? 1 : count + 1); } for (Entry<String, Integer> entry : result.entrySet()) { System.out.println(entry.getKey() + ", count=" + entry.getValue() +", reate="+ entry.getValue()/10000d); } } }
測(cè)試結(jié)果
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
IDEA2020.1創(chuàng)建springboot項(xiàng)目(國(guó)內(nèi)腳手架)安裝lombok
這篇文章主要介紹了IDEA2020.1創(chuàng)建springboot項(xiàng)目(國(guó)內(nèi)腳手架)安裝lombok,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06詳談java中File類getPath()、getAbsolutePath()、getCanonical的區(qū)別
下面小編就為大家?guī)?lái)一篇詳談java中File類getPath()、getAbsolutePath()、getCanonical的區(qū)別。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-07-07Java字符流和字節(jié)流對(duì)文件操作的區(qū)別
本篇文章主要介紹了Java的IO流分為字符流(Reader,Writer)和字節(jié)流(InputStream,OutputStream),字節(jié)流顧名思義字節(jié)流就是將文件的內(nèi)容讀取到字節(jié)數(shù)組,對(duì)初學(xué)者很有用,有需要的朋友可以了解一下。2016-10-10詳解Spring Batch 輕量級(jí)批處理框架實(shí)踐
這篇文章主要介紹了詳解Spring Batch 輕量級(jí)批處理框架實(shí)踐,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06詳解用Kotlin寫一個(gè)基于Spring Boot的RESTful服務(wù)
這篇文章主要介紹了詳解用Kotlin寫一個(gè)基于Spring Boot的RESTful服務(wù) ,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-05-05聊聊SpringCloud中的Ribbon進(jìn)行服務(wù)調(diào)用的問(wèn)題
SpringCloud-Ribbon是基于Netflix?Ribbon實(shí)現(xiàn)的一套客戶端負(fù)載均衡的工具。本文給大家介紹SpringCloud中的Ribbon進(jìn)行服務(wù)調(diào)用的問(wèn)題,感興趣的朋友跟隨小編一起看看吧2022-01-01springbean的加載過(guò)程及應(yīng)用場(chǎng)景分析
Bean已經(jīng)被實(shí)例化、屬性注入、初始化,并且注冊(cè)到容器中,可以被其他Bean或應(yīng)用程序使用,這篇文章主要介紹了springbean的加載過(guò)程以及應(yīng)用場(chǎng)景,需要的朋友可以參考下2024-04-04SpringBoot停止啟動(dòng)時(shí)測(cè)試檢查rabbitmq操作
這篇文章主要介紹了SpringBoot停止啟動(dòng)時(shí)測(cè)試檢查rabbitmq操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09JAVA實(shí)現(xiàn)JSON后端向前端傳遞數(shù)據(jù)
本篇文章主要介紹了JAVA實(shí)現(xiàn)JSON后端向前端傳遞數(shù)據(jù),這里整理了詳細(xì)的代碼,具有一定的參考價(jià)值,有需要的小伙伴可以參考下。2017-03-03SpringBoot集成Redisson實(shí)現(xiàn)消息隊(duì)列的示例代碼
本文介紹了如何在SpringBoot中通過(guò)集成Redisson來(lái)實(shí)現(xiàn)消息隊(duì)列的功能,包括RedisQueue、RedisQueueInit、RedisQueueListener、RedisQueueService等相關(guān)組件的實(shí)現(xiàn)和測(cè)試,感興趣的可以了解一下2024-10-10