紅包分配問題

来源:https://www.cnblogs.com/FangwayLee/archive/2023/07/11/17545782.html
-Advertisement-
Play Games

紅包分配問題 給你一個整數表示紅包的總額,和另一個整數表示紅包的個數 表示我們要把總金額,隨機分成N個紅包。 要求1:每個紅包的金額都是隨機的 要求2:每個人至少1分錢 示例代碼: 1 public class Test2 { 2 public static void main(String[] a ...


紅包分配問題

給你一個整數表示紅包的總額,和另一個整數表示紅包的個數

表示我們要把總金額,隨機分成N個紅包。

要求1:每個紅包的金額都是隨機的

要求2:每個人至少1分錢

 

示例代碼:

 1 public class Test2 {
 2     public static void main(String[] args) {
 3         System.out.println(Arrays.toString(luckyMoney("150.01", 10)));
 4     }
 5 
 6     public static BigDecimal[] luckyMoney(String money,int n){//紅包金額/元,紅包個數
 7         double money1= 0;
 8         try {
 9             money1 = Double.parseDouble(money);
10         } catch (NumberFormatException e) {
11             System.out.println("輸入金額錯誤!");
12             e.printStackTrace();
13             return null;
14         }
15         //System.out.println(money1);
16         int total=(int) (money1*100);//因為人民幣的金額最小擋位為分,可以先將紅包的金額轉化為以分為單位,轉化成整型
17         int rest=total-n;//紅包剩餘金額,因為每個紅包的最小金額為1分,所以可以把每個紅包的初始量設為1分
18         int[] luckMoney=new int[n];
19         Random r = new Random();
20         for (int i = 0; i < luckMoney.length; i++) {
21             if (i==luckMoney.length-1){//最後一個紅包金額等於紅包金額的最後剩餘量+1分
22                 luckMoney[luckMoney.length-1]=1+rest;
23                 break;
24             }
25             if (rest>0){
26                 int temp=r.nextInt(rest+1);//
27                 luckMoney[i]=1+temp;
28                 rest-=temp;
29             }else {//紅包剩餘金額已經分配完了,剩餘紅包的金額都為1分
30                 luckMoney[i]=1;
31             }
32         }
33         BigDecimal[] luckMoney1=new BigDecimal[n];//這裡用BigDecimal數組來記錄紅包的金額,因為BigDecimal進行計算時不會丟失精度
34         //BigDecimal sum = new BigDecimal("0");
35         for (int i = 0; i < luckMoney.length; i++) {
36             BigDecimal a = new BigDecimal(""+luckMoney[i]);
37             BigDecimal b = new BigDecimal("100");
38             luckMoney1[i]= a.divide(b,2,BigDecimal.ROUND_HALF_UP);//轉化以元為單位時,除以100,應保留兩位小數
39             sum=sum.add(luckMoney1[i]);
40         }
41         //System.out.println(sum);
42         return luckMoney1;
43     }
44 }

運行結果:

[16.50, 19.03, 64.12, 30.99, 8.96, 1.57, 2.10, 5.22, 0.04, 1.48]
[17.18, 112.52, 5.46, 0.97, 6.43, 3.26, 1.88, 1.27, 0.95, 0.09]
[82.38, 23.82, 14.27, 18.89, 4.31, 1.34, 4.16, 0.34, 0.12, 0.38]

這個演算法雖然能達到隨機分配紅包金額的功能,但由上面運行結果我們不難發現,越往後紅包分配的可金額越小,而且紅包分配不夠均勻

為了保證每個紅包金額分配額度的合理,

額度應該在0.01和剩餘平均值×2之間。例如:發100塊錢,總共10個紅包,那麼平均值是10塊錢一個,那麼發出來的紅包的額度在0.01元~20元之間波動。
當前面3個紅包總共被領了40塊錢時,剩下60塊錢,總共7個紅包,那麼這7個紅包的額度在:0.01~(60/7×2)=17.14之間。 修改後的代碼為
 1 public class Test2 {
 2     public static void main(String[] args) {
 3         System.out.println(Arrays.toString(luckyMoney("150.01", 10)));
 4     }
 5 
 6     public static BigDecimal[] luckyMoney(String money,int n){
 7         double money1= 0;
 8         try {
 9             money1 = Double.parseDouble(money);
10         } catch (NumberFormatException e) {
11             System.out.println("輸入金額錯誤!");
12             e.printStackTrace();
13             return null;
14         }
15         System.out.println(money1);
16         int total=(int) (money1*100);
17         int rest=total-n;
18         int restNum=n;//剩餘紅包個數
19         int[] luckMoney=new int[n];
20         Random r = new Random();
21         for (int i = 0; i < luckMoney.length; i++) {
22             if (i==luckMoney.length-1){
23                 luckMoney[luckMoney.length-1]=1+rest;
24                 break;
25             }
26             if (rest>0){
27                 int temp=r.nextInt(rest*2/restNum+1);//修改處
28                 restNum--;
29                 luckMoney[i]=1+temp;
30                 rest-=temp;
31             }else {
32                 luckMoney[i]=1;
33             }
34         }
35         BigDecimal[] luckMoney1=new BigDecimal[n];
36         BigDecimal sum = new BigDecimal("0");
37         for (int i = 0; i < luckMoney.length; i++) {
38             BigDecimal a = new BigDecimal(""+luckMoney[i]);
39             BigDecimal b = new BigDecimal("100");
40             luckMoney1[i]= a.divide(b,2,BigDecimal.ROUND_HALF_UP);
41             sum=sum.add(luckMoney1[i]);
42         }
43         System.out.println(sum);
44         return luckMoney1;
45     }
46 }

運行結果:

[7.21, 30.82, 26.08, 21.54, 7.81, 22.19, 9.72, 15.15, 3.89, 5.60]

在紅包拆開之前,每個人,無論先後順序,搶到的紅包金額的數學期望都是一樣的,如果100元分成5個紅包,那麼每個人搶到的金額的數學期望就是20元,但有趣的是,雖然數學期望一樣,但概率密度卻有很大差別。如果想詳細瞭解紅包分配的問題可以閱讀這篇知乎文獻:微信紅包金額分配的演算法是怎樣的?誰比較容易到最佳手氣?誰有機會拿到較大的金額? - Jingchi Wang的回答 - 知乎 https://www.zhihu.com/question/28250396/answer/86302251




您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • >我們是[袋鼠雲數棧 UED 團隊](http://ued.dtstack.cn/),致力於打造優秀的一站式數據中台產品。我們始終保持工匠精神,探索前端道路,為社區積累並傳播經驗價值。。 >本文作者:修能 ***以下內容充滿個人觀點。◡ ヽ(`Д´)ノ ┻━┻*** # 前言 基於分佈表單的需求,在 ...
  • #### 模板模式的定義 其定義如下: Define the skeleton of an algorithm in an operation,deferring some steps to subclasses.Template Method lets subclasses redefine ce ...
  • ## 一:場景 20w的QPS的場景下,服務端架構應如何設計? ## 二:常規解決方案 可使用分散式緩存來抗,比如redis集群,6主6從,主提供讀寫,從作為備,不提供讀寫服務。1台平均抗3w併發,還可以抗住,如果QPS達到100w,通過增加redis集群中的機器數量,可以擴展緩存的容量和併發讀寫能 ...
  • [toc] # 一、爬取目標 您好,我是[@馬哥python說](https://www.zhihu.com/people/13273183132),一名10年程式猿。 本次爬取的目標是: [微博熱搜榜](https://s.weibo.com/top/summary?cate=realtimeho ...
  • 與java類似,python作為一種面向對象的編程語言,也可以創建自定義的對象和類。 它的特性主要有:繼承,封裝,多態,方法,屬性,超類 # 1.變數的作用域 ```python c = 50 #全局變數, 作用域為整個模塊,若被引用,可作用域整個包 def plus(x,y): c = x + y ...
  • [toc] # 一、爬取目標 您好,我是[@馬哥python說](https://www.zhihu.com/people/13273183132),一名10年程式猿。 本次爬取的目標是:[百度熱搜榜](https://top.baidu.com/board?tab=realtime) ![百度熱搜 ...
  • 一. 介紹 值傳遞:值傳遞(Pass-by-Value)當我們向方法傳遞參數時,實際上是將該參數的值進行拷貝,並將拷貝後的值傳遞給方法內部。在方法內部對參數進行修改不會影響原始變數的值。 引用傳遞:引用傳遞(Pass by reference)是指方法調用時實參(即傳入方法的參數)是一個對象的引用, ...
  • # 2022藍橋杯b組 ## A題 ![img](https://img-blog.csdnimg.cn/4ac63a09fe784d7a94b710fc0cc48d09.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50, ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...