Java实现按比抽奖功能
作者:喵喵侠客 发布时间:2023-11-11 13:12:30
标签:java,抽奖
需求是要做几个小游戏的抽奖功能,需要根据不同的游戏有不同的抽奖规则,其中也有很多共性,可归纳为只按奖品占比抽取、奖品占比与奖品数量抽取、分段抽取,为方便起见将这些的抽奖的规则统一封装到了工具类中。抽奖的核心逻辑使用的叫做离散算法实现的。
一.概述
使用离散算法即根据奖品占比进行分段,然后再产生随机数匹配所对应的区间。
首先定义Prize奖品实体类,类中有prizeName(奖品名称)、prizeWeight(奖品比重)、prizeCount(奖品数量)属性,下面是核心的代码:
/**
* 按比例随机抽取一项
* @param list 奖品列表
* @return 类型值
*/
public static String ratioExtract(List<Prize> list) {
//非空判断
if (list==null || list.size()<1) {
return null;
}
//占比之和
double sum=0.00;
//分段数组(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;
/* 产生随机数 */
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;
}
二、测试
以下是完整的抽奖工具类
import lombok.Data;
import org.apache.commons.lang.math.RandomUtils;
import java.util.List;
import java.util.Random;
/**
* @Description: 抽奖工具类
* @author: xiake
* @Date: 2020/1/5 13:23
* @ModifiedDate:2020/1/5 13:23
* @Copyright: miaoxaike.com
*/
public class PrizeMathRandom {
/**
* 按比例随机抽取一项
* @param fieldArray 类型值数组
* @param proportions 与类型值对应 的占比值
* @return 类型值
*/
public static String ratioExtract(String[] fieldArray,double[] proportions) {
//判断两个数组长度是否相等
if(fieldArray.length!=proportions.length) {
return "两数组长度不相等,无法执行";
}
//占比之和
double sum=0.00;
//分段数组(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();
/* 产生随机数 区间为 (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;
}
/**
* 按比例随机抽取一项
* @param list 奖品列表
* @return 类型值
*/
public static String ratioExtract(List<Prize> list) {
//非空判断
if (list==null || list.size()<1) {
return null;
}
//占比之和
double sum=0.00;
//分段数组(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;
/* 产生随机数 */
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 分段数组, 参数值用"-"组装(例: {"6-14","14-23","23-32","32-40"})
* @param proportions 每段出现的概率
* @return 返回按比例抽取后, 分段范围内的随机一个值
*/
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{
//奖品名称
private String prizeName;
//奖品占比
private double prizeWeight;
//奖品数量
private int prizeCount;
}
}
除了核心的实现方法外另外还补充了两个扩充的方法为满足游戏规则所用。下面简单做个测试
public static void main(String[] args) {
//初始化奖品信息
List<Prize> prizeList=new ArrayList<>();
prizeList.add(new Prize("一等奖",1,1));
prizeList.add(new Prize("二等奖",3,4));
prizeList.add(new Prize("三等奖",6,5));
for (int i = 0; i < 12; i++) {
Prize prize = ratioExtract(prizeList);
if (prize!=null){
System.out.println("第"+(i+1)+"次,抽中 "+prize.getPrizeName()+" 剩余奖品数量="+prize.getPrizeCount());
}else {
System.out.println("第"+(i+1)+"次,奖品已抽完");
}
}
}
运行效果如下
实现的方法很简单,可能还有些不合理的地方,但也足以满足当前需求了。基本上都是对数组与随机数的使用就不详细讲解了,有问题欢迎在评论区留言!
来源:https://blog.csdn.net/qq_38217237/article/details/103836432


猜你喜欢
- 实践过程效果代码public partial class Form1 : Form{ public Form1()
- 在上周发布的 TienChin 项目视频中,我和大家一共梳理了六种幂等性解决方案,接口幂等性处理算是一个非常常见的需求了,我们在很多项目中其
- 目录概述非阻塞算法依赖JCTools队列队列实现原子队列容量其他数据结构工具性能测试使用JCTools的缺点结论概述在本文中,我们将介绍JC
- 我们在学习接口的时候。能够在里面做一些方法的调用。不过今天所要讲的JDBC,虽然也是连接数据库的一种接口,不过与类接口有着很大的区别,大家要
- 实践过程效果代码public partial class Form1 : Form{ public Form1()
- 前言在Android开发中,View一直是Android开发人员的一块心病,一方面想要进阶,一方面又害怕进阶,可以说Android的View
- 1.现象描述原来项目在Android studio 2.3一切正常,升级3.0之后报如下错误:Error:Cannot choose bet
- 前言对于Android注解,或多或少都有一点接触,但相信大多数人都是在使用其它依赖库的时候接触的。因为有些库如果你想使用它就必须使用它所提供
- 快速阅读如何在winform程序中,让界面不再卡死。 关于委托和AsyncCallback的使用。界面卡死的原因是因为耗时任务的计算占用了主
- 摘要 想必大家对小榕时光等扫描器都非常熟悉了,有没有自己写一个的冲动。最近微软推实施了.NET战略方案,C#是主推语言,你们是否
- MybatisPlus分页排序查询字段带有下划线如果使用MybatisPlus的自动转驼峰命名法,分页排序查询的字段带有下划线时,会出问题。
- 前言Spring框架对Bean进行装配提供了很灵活的方式,下面归纳一下主要的方式:在XML中进行显示配置在Java中进行显示配置隐式的bea
- Spring5路径匹配器PathPatternPathPattern 对url地址匹配的处理更加快速,它和AntPathMatcher 主要
- 前言最近做公司项目的时候,经常会遇到一个问题,就是我为某个控件如EditText设置requestfocus()的时候不管用,比如说登陆的时
- 引言近期,Google 发布了 Android 11的平台稳定版本。Android 11 将在确保用户隐私安全的前提下,更好地让用户畅享最新
- 对于静态变量、静态初始化块、变量、初始化块、构造器,它们的初始化顺序依次是(静态变量、静态初始化块)>(变量、初始化块)>构造器
- 1. 为什么使用线程池诸如 Web 服务器、数据库服务器、文件服务器或邮件服务器之类的许多服务器应用程序都面向处理来自某些远程来源的大量短小
- 指定创建派生类实例时应调用的基类构造函数;调用基类上已被其他方法重写的方法。注意:不能从静态方法中使用base关键字,base关键字只能在实
- 本文实例为大家分享了Javafx实现国际象棋游戏的具体代码,供大家参考,具体内容如下基本规则棋子马设计“日”的移动方式兵设计只能向前直走,每
- 这段时间用到了QT的TCP通信,做了初步的学习与尝试,编写了一个客户端和服务器基于窗口通信的小例程。使用QT的网络套接字需要.pro文件中加