wx.checkjsapi是寫在config里面嗎_用Python寫一個程序,解密游戲內抽獎的秘密

af0fcc30b540573985aa904098ba21b6.png

前言

本文的文字及圖片來源于網絡,僅供學習、交流使用,不具有任何商業用途,版權歸原作者所有,如有問題請及時聯系我們以作處理。

作者: 極客挖掘機

PS:如有需要Python學習資料的小伙伴可以加點擊下方鏈接自行獲取

http://t.cn/A6Zvjdun

分析需求

我們先整理下思路,目標是什么? 目標是要寫一個抽獎程序,那么抽獎程序的核心是什么? 當然是如何判斷一個人中獎了。那么如何判斷一個人中獎呢? 是不是可以通過隨機函數來操作呢?

中獎方法

一步一步來,我們先通過隨機函數來判斷是否中獎。代碼是不是可以先寫成下面這樣:

import random# 判斷中獎函數def lottery(): flag = random.randint(0, 9) if flag < 2: return True else: return False

首先,我們獲取 0 ~ 9 之間的隨機正整數(這里不討論 random 是不是很隨機,從狹義上來講我們可以認為它是隨機的),如果中獎率為 20% 的話,我們可以認為小于 2 的數字為中獎,其余的為沒有中獎。然后中獎后返回 True ,沒有中獎返回 False 。

我們加一個入口測試函數,測試一下上面的代碼是否能正常運行,并且中獎率是否能維持在大約 20 % 左右。

if __name__ == '__main__': # 中獎次數 a = 0 # 沒有中獎次數 b = 0 for i in range(1000000) : if (lottery()): a += 1 else: b += 1 print('共計中獎:', a, ',未中獎:', b)

執行結果:

共計中獎: 200145 ,未中獎: 799855

上面的測試總共循環了 1 百萬次,大約執行需要 2 ~ 3 秒左右,速度還是蠻快的。可以看到,中獎結果確實接近 20% 左右。

動態中獎率

難道到這里就結束了么?當然不可能,這里只是剛剛開了個頭。

如果這時老板說,你這個概率不能調整啊,需要讓中獎率可以動態調整的,活動剛開始的時候中獎率要高,隨著時間的推移,中獎率要降下來。

這時候咋整,傻眼了吧。

既然中獎率要可調整,那么我們中獎率就不能定死在程序中了,這個中獎率需要有一個地方去做存儲,在每次做隨機的時候將這個中獎率取出來。

簡單易行的方法就是將這個中獎率放在數據庫中或者緩存服務中,這個根據實際業務場景來定。一般是根據預估訪問壓力的大小來進行技術選型,如果壓力不是特別大,那么放在數據庫中也是可以的,如果并發會比較高的話,建議還是放在緩存中。

我們來寫一個從數據庫獲取中獎概率的方法(為了展示直觀,小編這里直接使用 Mysql 數據庫用作數據存儲),先看下數據庫的數據:

1e4ebfef1b26891243a90d0a2717def4.png

很簡單的設計了一張表,里面有意義的字段有兩個,一個用作中獎率的分子部分,一個用作中獎率的分母部分。分母部分最好要設置成 100 、 1000 、 10000 這種,這樣計算中獎率會比較好計算。

def get_lottery_rate(): conn = pymysql.connect(host='localhost', user='root', password='password', database='test', charset='utf8mb4') try: sql = 'SELECT fenzi, fenmu FROM rate' cursor = conn.cursor() cursor.execute(sql) result = cursor.fetchone() return result except Exception as ex: print(ex) finally: conn.close()

運行這個方法測試結果如下:

(10, 100)

可以看到,我們獲得了一個元組,里面的內容就是我們從數據庫取出來的分子和分母。

我們將前面的抽獎的那個方法改一下,改成從數據庫獲取中獎比例。修改后的代碼如下:

def lottery(): rate = get_lottery_rate() flag = random.randint(1, rate[1]) if flag < rate[0]: return True else: return False

還是運行上面的測試方法,這里要注意下,因為我們現在是從數據庫獲取數據,每次方法執行都要加上數據庫鏈接的建立與銷毀,建議將循環次數修改為 1000 以內,不然執行的時間就有點太長了。

小編這里將循環次數修改為 1000 次后,執行結果如下:

共計中獎: 92 ,未中獎: 908

那么到這里,我們就可以通過修改數據庫中數據實時的操作中獎率了。當然上面的慢的問題我們可以使用數據庫連接池等技術進行優化。

增加獎項

那么是否就結束了呢?no no no,我們接著加需求。

現在,我們只能知道每次到底中不中獎,只有一個獎項,但是現在想變成 3 個獎項,如:一等獎、二等獎、三等獎那該怎么辦?

這個對之前的抽獎方法改動就有點大了,首先我們先在數據庫增加出來另外兩個獎項的配置:

1997855afeae02cf027008433d37b33a.png

配置這里三個獎項的分母最好保持一致,否則后續計算會徒增復雜度。

修改我們獲取配置的那個方法:

def get_lottery_rate(): conn = pymysql.connect(host='localhost', port = 3306, user='root', password='password', database='test', charset='utf8mb4') try: sql = 'SELECT * FROM rate order by id asc ' cursor = conn.cursor() cursor.execute(sql) result = cursor.fetchall() return result except Exception as ex: print(ex) finally: conn.close()

測試調用后結果如下:

((1, 10, 100), (2, 5, 100), (3, 1, 100))

先在我們要做的是要將這個配置融入進我們之前的中獎的那個方法中,不多說,直接上代碼:

# 判斷中獎函數def lottery(): config = get_lottery_rate() flag = random.randint(1, config[0][2]) if flag <= config[0][1]: return 1 elif flag > config[0][1] and flag <= (config[1][1] + config[0][1]): return 2 elif flag > (config[1][1] + config[0][1]) and flag <= (config[2][1] + config[1][1]): return 3 else: return 0

接著修改我們的做測試的代碼:

def main(): # 一等獎中獎次數 a = 0 # 二等獎中獎次數 b = 0 # 三等獎中獎次數 c = 0 # 未中獎次數 d = 0 # 循環次數 e = 0 for i in range(1000): e += 1 print('當前循環次數:', e) result = lottery() print('當前中獎結果:', result) if (result == 1): a += 1 elif (result == 2): b += 1 elif (result == 3): c += 1 else: d += 1 print('一等獎中獎:', a, ',二等獎中獎次數:', b, ',三等獎中獎次數:', c, ',未中獎次數:', d)

調用我們的測試方法:

if __name__ == '__main__': main()

小編這里的運行結果如下:

cc4dd773274d0dc4eb8a5791c629a68b.png

增加會員判斷

到這里我們還沒完,還能加需求,現在網站大多數都是會員制的,比如白銀會員,黃金會員,鉆石會員,如果不同的會員等級需要有不同的中獎率,這個是很正常的一件事兒,小編現在還清晰的記得當年某家大型互聯網公司代碼中的注釋 “窮逼 VIP(活動送的那種)” 。

我們假設鉆石會員的中獎率為整體中獎率的 100% ,黃金會員的中獎率為整體中獎率的 50% ,白銀會員的中獎率為整體中獎率的 20% 。

最簡單的實現方式是直接在最外層套一層會員中獎率的判斷,不知道各位同學怎么想。

小編這里給出自己的解決方案:

# 判斷會員等級中獎率過濾# 會員等級 1.白銀會員 2.黃金會員 3. 鉆石會員def vip_lottery(level): rate = random.randint(1, 10) # 如果是鉆石會員,直接進入抽獎函數 if level == 3: return lottery() # 如果是黃金會員, 50% 概率進入抽獎函數 elif level == 2: if rate <= 5: return lottery() else: return 0 # 如果是白銀會員, 20% 概率進入抽獎函數 elif level == 1: if rate <= 2: return lottery() else: return 0 # 如果是其他,直接返回未中獎 else: return 0

我們新增一個測試增加會員過濾的測試方法:

# 會員制中獎測試方法def test_vip(): print('請輸入您當前的會員等級:1.白銀會員 2.黃金會員 3. 鉆石會員') level = input() result = vip_lottery(int(level)) if (result == 1): print('恭喜您中了一等獎') elif (result == 2): print('恭喜您中了二等獎') elif (result == 3): print('恭喜您中了三等獎') else: print('未中獎,謝謝惠顧')

在我們的入口函數中調用這個方法:

if __name__ == '__main__': test_vip()

最終測試結果如下:

357da79df584cf88a8476f5863a368a6.png

小編的人品還可以嘛,直接就能中三等獎。

import randomimport pymysql# 獲取中獎配置def get_lottery_rate(): conn = pymysql.connect(host='114.67.111.196', port = 3306, user='root', password='wsy@123456', database='test', charset='utf8mb4') try: sql = 'SELECT * FROM rate order by id asc ' cursor = conn.cursor() cursor.execute(sql) result = cursor.fetchall() return result except Exception as ex: print(ex) finally: conn.close()# 判斷中獎函數def lottery(): config = get_lottery_rate() flag = random.randint(1, config[0][2]) if flag <= config[0][1]: return 1 elif flag > config[0][1] and flag <= (config[1][1] + config[0][1]): return 2 elif flag > (config[1][1] + config[0][1]) and flag <= (config[2][1] + config[1][1]): return 3 else: return 0# 判斷會員等級中獎率過濾# 會員等級 1.白銀會員 2.黃金會員 3. 鉆石會員def vip_lottery(level): rate = random.randint(1, 10) # 如果是鉆石會員,直接進入抽獎函數 if level == 3: return lottery() # 如果是黃金會員, 50% 概率進入抽獎函數 elif level == 2: if rate <= 5: return lottery() else: return 0 # 如果是白銀會員, 20% 概率進入抽獎函數 elif level == 1: if rate <= 2: return lottery() else: return 0 # 如果是其他,直接返回未中獎 else: return 0# 批量測試方法def test(): # 一等獎中獎次數 a = 0 # 二等獎中獎次數 b = 0 # 三等獎中獎次數 c = 0 # 未中獎次數 d = 0 # 循環次數 e = 0 for i in range(1000): e += 1 print('當前循環次數:', e) result = lottery() print('當前中獎結果:', result) if (result == 1): a += 1 elif (result == 2): b += 1 elif (result == 3): c += 1 else: d += 1 print('一等獎中獎:', a, ',二等獎中獎次數:', b, ',三等獎中獎次數:', c, ',未中獎次數:', d)# 會員制中獎測試方法def test_vip(): print('請輸入您當前的會員等級:1.白銀會員 2.黃金會員 3. 鉆石會員') level = input() result = vip_lottery(int(level)) if (result == 1): print('恭喜您中了一等獎') elif (result == 2): print('恭喜您中了二等獎') elif (result == 3): print('恭喜您中了三等獎') else: print('未中獎,謝謝惠顧')if __name__ == '__main__': test_vip()

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/457813.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/457813.shtml
英文地址,請注明出處:http://en.pswp.cn/news/457813.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Dev C++安裝第三方庫boost

Dev_C安裝第三方庫boost 安裝步驟 準備工作下載boost庫&#xff0c;下載地址https://sourceforge.net/projects/boost/1. 設置GCC的環境變量PATH 設置環境變量path,在其中加上DEV-C編譯器的路徑&#xff08;gcc.exe所在路徑&#xff09;&#xff0c;如C:\Program Files (x86)…

bash的一些小技巧

1、從輸入讀入變量 eg:read -ep "input yes or no: " flag 用e選項表示編輯&#xff0c;可以使用backspace刪除 2、數組 a、索引數組 declare -a arr(var1 var2 var3) 用空格分割&#xff0c;如果直接訪問變量$arr&#xff0c; 則獲取的是數組的第一個元素&#xff0…

golang switch_為什么程序員都不喜歡使用 switch ,而是大量的 if……else if ?

點擊上方“我要學編程”&#xff0c;選擇“置頂/星標公眾號”福利干貨&#xff0c;第一時間送達&#xff01;來自 | C語言Plus請用5秒鐘的時間查看下面的代碼是否存在bug。OK&#xff0c;熟練的程序猿應該已經發現Bug所在了&#xff0c;在第13行下面我沒有添加關鍵字break; 這就…

RabbitMQ 安裝與簡單使用

在企業應用系統領域&#xff0c;會面對不同系統之間的通信、集成與整合&#xff0c;尤其當面臨異構系統時&#xff0c;這種分布式的調用與通信變得越發重要。其次&#xff0c;系統中一般會有很多對實時性要求不高的但是執行起來比較較耗時的地方&#xff0c;比如發送短信&#…

數據庫函數依賴及范式

一、基礎概念   要理解范式&#xff0c;首先必須對知道什么是關系數據庫&#xff0c;如果你不知道&#xff0c;我可以簡單的不能再簡單的說一下&#xff1a;關系數據庫就是用二維表來保存數據。表和表之間可以……&#xff08;省略10W字&#xff09;。   然后你應該理解以下…

windows svn

windows svn 1.1Svn和VisualSvn介紹 VisualSvn Server2.5.6&#xff08;版本控制服務器&#xff09;免費開源軟件 是基于Windows平臺上的Subversion服務器&#xff0c;它是免費的 官方下載&#xff1a; http://www.visualsvn.com/files/VisualSVN-Server-2.5.6.msi TortoiseSvn…

信息摘要技術及算法介紹

數據摘要算法是密碼學算法中非常重要的一個分支&#xff0c;它通過對所有數據提取指紋信息以實現數據簽名、數據完整性校驗等功能&#xff0c;由于其不可逆性&#xff0c;有時候會被用做敏感信息的加密。 數據摘要算法也被稱為哈希&#xff08;Hash&#xff09;算法、散列算法…

AutoLayout的那些事兒

AutoLayout非常強大也非常易用&#xff0c;可讀性也很強&#xff0c;加上各種第三方AutoLayout庫&#xff0c;讓你布起局來猶如繃掉鏈子的狗&#xff01;根本停不下來&#xff01;以前的 1label.frame.origin.y label.frame.size.height 10如今只用&#xff1a; 123button.sn…

docker-compose下載慢_編寫Docker Compose時要注意的五大常見錯誤

在構建容器化的應用時&#xff0c;開發人員往往需要某種方法來引導啟動目標容器&#xff0c;以對其進行代碼級別的測試。盡管業界有許多方法可以實現該目的&#xff0c;但Docker Compose是目前最受歡迎的一種方法。它能夠讓如下兩個方面變得容易實現&#xff1a;指定在開發過程…

前端測試利器--Browser-Sync啟動命令

使用browser-sync啟動命令cmd切換到項目的根目錄下**1.browser-sync start --server --files "css/*.css"----------**使用兩個*檢測所有的目錄**轉載于:https://blog.51cto.com/1888512/1862054

VMware實現Android x86 8.1 從安裝到使用

VMware實現Android x86 8.1 從安裝到使用 虛擬機--Android 安裝 Android系統配置 安裝軟件 個性化設計 托坑指南 一些終端模擬器的指令 虛擬機–Android 發現現在安卓虛擬機已經到了8.1&#xff0c;我就試試能不能安裝并正常使用。由于版本過新&#xff0c;網上也沒有一些系統的…

frame越過另一個frame_擁抱swoole(三)之用php實現一個混合服務器

混合服務器&#xff0c;就是可以同時支持http&#xff0c;websocket&#xff0c;tcp等的服務器&#xff0c;用swoole就是這么簡單&#xff0c;分分鐘&#xff0c;就可以愉快地搞物聯網開發了&#xff0c;啥都支持&#xff0c;我采用官方的例子&#xff0c;創建一個混合服務器&a…

Hibernate學習系列————注解一對多單向實例

2019獨角獸企業重金招聘Python工程師標準>>> 開發環境&#xff1a;MysqlEclipse 一對多單向的列子原理&#xff1a;一個班級&#xff0c;多個學生&#xff0c;學生端為多的一端&#xff0c;他們擁有一個外鍵指向相同的班級。 項目結構 需要的jar包 hibernate.cfg.xm…

Spring學習筆記--自動裝配Bean屬性

Spring提供了四種類型的自動裝配策略&#xff1a; byName – 把與Bean的屬性具有相同名字(或者ID)的其他Bean自動裝配到Bean的對應屬性中。byType – 把與Bean的屬性具有相同類型的其他Bean自動裝配到Bean的對應屬性中。constructor – 把與Bean的構造器入參具有相同類型的其他…

sudo apt-get nmap 報錯鎖占用

在Ubuntu中用apt-get命令安裝軟件是出現如下錯誤&#xff1a; 網上搜了一下原因&#xff0c;說是有另外一個程序在運行&#xff0c;導致鎖不可用&#xff0c;原因可能是賞析運行更新或安裝沒有正常完成。這是因為上次更新或者安裝沒有正常完成。 網上的兩種解決方法&#xff1…

python逐行讀取txt寫入excel_用python從符合一定格式的txt文檔中逐行讀取數據并按一定規則寫入excel(openpyxl支持Excel 2007 .xlsx格式)...

前幾天接到一個任務&#xff0c;從gerrit上通過ssh命令獲取一些commit相關的數據到文本文檔中&#xff0c;隨后將這些數據存入Excel中。數據格式如下圖所示觀察上圖可知&#xff0c;存在文本文檔中的數據符合一定的格式&#xff0c;通過python讀取、正則表達式處理并寫入Excel文…

筋斗云newcloud錯誤碼列表

響應碼信息備注440Ip Error客戶送IP錯誤441Callee Number Error被叫號碼位數錯誤&#xff08;標準11位正確&#xff0c;錯誤加前綴0&#xff0c;或其他前綴&#xff09;442Called Operator Error被叫運營商錯誤&#xff08;支持移動&#xff0c;不支持聯通電信&#xff09;443N…

Extjs 之 initComponent 和 constructor的區別(轉)

在創建自定義類時&#xff0c;先構造&#xff08;constructor)后初始化&#xff08;initComponent&#xff09;。如&#xff1a;&#xff08;在舊的Extjs 版本中使用 Ext.extend 實現擴展&#xff09; Ext.define(Btn,{ extend:Ext.button.Button, init…

hive遍歷_從Hive中的stored as file_foramt看hive調優

一、行式數據庫和列式數據庫的對比1、存儲比較行式數據庫存儲在hdfs上式按行進行存儲的&#xff0c;一個block存儲一或多行數據。而列式數據庫在hdfs上則是按照列進行存儲&#xff0c;一個block可能有一列或多列數據。2、壓縮比較對于行式數據庫&#xff0c;必然按行壓縮&#…

oracle sql語句 從指定條數查詢

現有表A 查詢從第10行之后的數據 select a from ( select a, rownum r from A ) where r > 10 order by r; 實際工作中例子 select account,acct_name from ( select account, acct_name, rownum r from pmctl_nonsleep_acct ) where r > 10 order by