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ù)匹配所對應(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;
  }
二、測試
以下是完整的抽獎(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 與類型值對應(yīng) 的占比值
   * @return 類型值
   */
  public static String ratioExtract(String[] fieldArray,double[] proportions) {
    //判斷兩個(gè)數(shù)組長度是否相等
    if(fieldArray.length!=proportions.length) {
      return "兩數(shù)組長度不相等,無法執(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ī)則所用。下面簡單做個(gè)測試
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)的方法很簡單,可能還有些不合理的地方,但也足以滿足當(dāng)前需求了?;旧隙际菍?shù)組與隨機(jī)數(shù)的使用就不詳細(xì)講解了,有問題歡迎在評論區(qū)留言!
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
上一篇:Spring Boot Debug調(diào)試過程圖解
欄 目:Java
本文標(biāo)題:Java實(shí)現(xiàn)按比抽獎(jiǎng)功能
本文地址:http://www.jygsgssxh.com/a1/Java/8723.html
您可能感興趣的文章
- 01-10Java實(shí)現(xiàn)動(dòng)態(tài)模擬時(shí)鐘
 - 01-10利用Java實(shí)現(xiàn)復(fù)制Excel工作表功能
 - 01-10JavaWeb實(shí)現(xiàn)郵件發(fā)送功能
 - 01-10java基于poi導(dǎo)出excel透視表代碼實(shí)例
 - 01-10Java實(shí)現(xiàn)動(dòng)態(tài)數(shù)字時(shí)鐘
 - 01-10基于Java驗(yàn)證jwt token代碼實(shí)例
 - 01-10java實(shí)現(xiàn)液晶數(shù)字字體顯示當(dāng)前時(shí)間
 - 01-10淺談Java中真的只有值傳遞么
 - 01-10Java動(dòng)態(tài)顯示當(dāng)前日期和時(shí)間
 - 01-10如何解決線程太多導(dǎo)致java socket連接池出現(xiàn)的問題
 


閱讀排行
本欄相關(guān)
- 01-10Java實(shí)現(xiàn)動(dòng)態(tài)模擬時(shí)鐘
 - 01-10Springboot中@Value的使用詳解
 - 01-10JavaWeb實(shí)現(xiàn)郵件發(fā)送功能
 - 01-10利用Java實(shí)現(xiàn)復(fù)制Excel工作表功能
 - 01-10Java實(shí)現(xiàn)動(dòng)態(tài)數(shù)字時(shí)鐘
 - 01-10java基于poi導(dǎo)出excel透視表代碼實(shí)例
 - 01-10java實(shí)現(xiàn)液晶數(shù)字字體顯示當(dāng)前時(shí)間
 - 01-10基于Java驗(yàn)證jwt token代碼實(shí)例
 - 01-10Java動(dòng)態(tài)顯示當(dāng)前日期和時(shí)間
 - 01-10淺談Java中真的只有值傳遞么
 
隨機(jī)閱讀
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
 - 04-02jquery與jsp,用jquery
 - 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
 - 01-10C#中split用法實(shí)例總結(jié)
 - 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
 - 01-10delphi制作wav文件的方法
 - 08-05DEDE織夢data目錄下的sessions文件夾有什
 - 01-11ajax實(shí)現(xiàn)頁面的局部加載
 - 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
 - 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
 


