Java實(shí)現(xiàn)按比抽獎(jiǎng)功能
需求是要做幾個(gè)小游戲的抽獎(jiǎng)功能,需要根據(jù)不同的游戲有不同的抽獎(jiǎng)規(guī)則,其中也有很多共性,可歸納為只按獎(jiǎng)品占比抽取、獎(jiǎng)品占比與獎(jiǎng)品數(shù)量抽取、分段抽取,為方便起見(jiàn)將這些的抽獎(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)度不相等,無(wú)法執(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ì)講解了,有問(wèn)題歡迎在評(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)盤(pán)抽獎(jiǎng)Java服務(wù)端接口詳解
- java實(shí)現(xiàn)幸運(yùn)抽獎(jiǎng)系統(tǒng)
- java實(shí)現(xiàn)抽獎(jiǎng)功能解析
相關(guān)文章
SpringMVC xml文件路徑在web.xml中的配置方式
這篇文章主要介紹了SpringMVC xml文件路徑在web.xml中的配置方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
MyBatis分頁(yè)插件PageHelper的使用與原理
提到插件相信大家都知道,插件的存在主要是用來(lái)改變或者增強(qiáng)原有的功能,MyBatis中也一樣,下面這篇文章主要給大家介紹了關(guān)于Mybatis第三方PageHelper分頁(yè)插件的使用與原理,需要的朋友可以參考下2023-02-02
SpringBoot項(xiàng)目Jar包如何瘦身部署的實(shí)現(xiàn)
這篇文章主要介紹了SpringBoot項(xiàng)目Jar包如何瘦身部署的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09
Java使用NIO包實(shí)現(xiàn)Socket通信的實(shí)例代碼
本篇文章主要介紹了Java使用NIO包實(shí)現(xiàn)Socket通信的實(shí)例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-02-02

