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í)體類(lèi)
*/
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)文章
詳解DES加密算法的原理與Java實(shí)現(xiàn)
DES 加密,是對(duì)稱(chēng)加密。對(duì)稱(chēng)加密,顧名思義,加密和解密的運(yùn)算全都是使用的同樣的秘鑰。這篇文章主要為大家講講DES加密算法的原理與Java實(shí)現(xiàn),需要的可以參考一下2022-10-10
Java中l(wèi)ombok的@Builder注解的解析與簡(jiǎn)單使用詳解
這篇文章主要介紹了Java中l(wèi)ombok的@Builder注解的解析與簡(jiǎn)單使用,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01
淺談Java中BigDecimal類(lèi)的簡(jiǎn)單應(yīng)用
這篇文章主要介紹了淺談Java中BigDecimal類(lèi)的簡(jiǎn)單應(yīng)用,BigDecimal是由任意精度的整數(shù)非標(biāo)度值和32位的整數(shù)標(biāo)度組成,如果為零或正數(shù),則標(biāo)度是小數(shù)點(diǎn)后的位數(shù),如果為負(fù)數(shù),則將該數(shù)的非標(biāo)度值乘以?10的負(fù)scale次冪,需要的朋友可以參考下2023-07-07
Spring?Data?Elasticsearch使用方式(Elasticsearch)
這篇文章主要介紹了Spring?Data?Elasticsearch使用方式(Elasticsearch),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2025-04-04
springboot調(diào)用python文件的詳細(xì)方案
這篇文章主要為大家詳細(xì)介紹了springboot調(diào)用python文件的詳細(xì)方案,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2025-04-04
Springboot基于websocket實(shí)現(xiàn)簡(jiǎn)單在線聊天功能
這篇文章主要介紹了Springboot基于websocket實(shí)現(xiàn)簡(jiǎn)單在線聊天功能,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-06-06

