JavaScript代碼實現(xiàn)春晚劉謙魔術的模擬程序
什么是約瑟夫環(huán)問題?
約瑟夫環(huán)(Josephus problem)是一個經典的數(shù)學問題,最早由古羅馬歷史學家弗拉維奧·約瑟夫斯提出,但它的名字是在19世紀由德國數(shù)學家約瑟夫·喬瑟夫斯(Josef Stein)命名的。
問題的描述是這樣的:假設有n個人(編號從1到n)站成一個圓圈,從第一個人開始報數(shù),報到某個數(shù)字(例如k)的人就被殺死,然后從下一個人開始重新報數(shù)并繼續(xù)這個過程,直到只剩下一個人留下來。
問題的關鍵是找出存活下來的那個人的編號。
結合撲克牌解釋約瑟夫環(huán)問題
1、考慮最簡單的情況
假設有2張牌,編號分別是1和2。
首先將1放到后面,扔掉2。剩下的就是最開始放在最上邊的那張1。
2、稍微復雜一點的情況,牌的張數(shù)是2的n次方
比如有8張牌,編號分別是1、2、3、4、5、6、7、8。
第一輪會把2、4、6、8扔掉,剩下1、3、5、7按順序放在后面,又退化成了4張牌的情況。
第二輪會把3、7扔掉,剩下1、5按順序放在后面,又退化成了2張牌的情況。
第三輪把5扔掉,剩下1,就是最初在最前面的那張。
結論:如果牌的張數(shù)是2^n,最后剩下的一定是最開始放在牌堆頂?shù)哪菑垺?/p>
3、考慮任意的情況,牌的張數(shù)是2^n+m
比如牌的張數(shù)是11,等于8+3。把1放到后面,把2扔掉,把3放到后面,把4扔掉,把5放到后面,把6扔掉,現(xiàn)在剩下的編號序列是7、8、9、10、11、1、3、5,這又是8張牌的情況!最后一定剩下的是現(xiàn)在牌堆頂?shù)?!
因此,只要提前知道牌的張數(shù),就一定能馬上推導出最終是剩下哪一張牌。一切的魔法都是數(shù)學?。《际撬惴ǎ?!
見證奇跡的時刻!魔術的流程
- 4張牌對折后撕開,就是8張,疊放在一起就是ABCDABCD。注意,ABCD四個數(shù)字是完全等價的。
- 根據(jù)名字字數(shù),把頂上的牌放到下面,但怎么放都不會改變循環(huán)序列的相對位置。譬如2次,最后變成CDABCDAB;譬如3次,最后換成DABCDABC。但無論怎么操作,第4張和第8張牌都是一樣的。
- 把頂上3張插到中間任意位置。這一步非常重要!因為操作完之后必然出現(xiàn)第1張和第8張牌是一樣的!以名字兩個字為例,可以寫成BxxxxxxB(這里的x是其他和B不同的牌)。
- 拿掉頂上的牌放到一邊,記為B。剩下的序列是xxxxxxB,一共7張牌。
- 南方人/北方人/不確定,分別拿頂上的1/2/3張牌插到中間,但是不會改變剩下7張牌是xxxxxxB的結果。
- 男生拿掉1張,女生拿掉2張。也就是男生剩下6張,女生剩下5張。分別是xxxxxB和xxxxB。
- 循環(huán)7次,把最頂上的放到最底下,男生和女生分別會是xxxxBx和xxBxx。
- 最后執(zhí)行約瑟夫環(huán)過程!操作到最后只剩下1張。當牌數(shù)為6時(男生),剩下的就是第5張牌;當牌數(shù)為5時(女生),剩下的就是第3張牌。Bingo!就是第4步拿掉的那張牌!
下面是完整的 JavaScript 代碼實現(xiàn):
// 定義一個函數(shù),用于把牌堆頂n張牌移動到末尾 function moveCardBack(n, arr) { // 循環(huán)n次,把隊列第一張牌放到隊列末尾 for (let i = 0; i < n; i++) { const moveCard = arr.shift(); // 彈出隊頭元素,即第一張牌 arr.push(moveCard); // 把原隊頭元素插入到序列末尾 } return arr; } // 定義一個函數(shù),用于把牌堆頂n張牌移動到中間的任意位置 function moveCardMiddleRandom(n, arr) { // 插入在arr中的的位置,隨機生成一個idx // 這個位置必須是在n+1到arr.length-1之間 const idx = Math.floor(Math.random() * (arr.length - n - 1)) + n + 1; // 執(zhí)行插入操作 const newArr = arr.slice(n, idx).concat(arr.slice(0, n)).concat(arr.slice(idx)); return newArr; } // 步驟1:初始化8張牌,假設為"ABCDABCD" let arr = ["A", "B", "C", "D", "A", "B", "C", "D"]; console.log("步驟1:拿出4張牌,對折撕成8張,按順序疊放。"); console.log("此時序列為:" + arr.join('') + "\n---"); // 步驟2(無關步驟):名字長度隨機選取,這里取2到5(其實任意整數(shù)都行) const nameLen = Math.floor(Math.random() * 4) + 2; // 把nameLen張牌移動到序列末尾 arr = moveCardBack(nameLen, arr); console.log(`步驟2:隨機選取名字長度為${nameLen},把第1張牌放到末尾,操作${nameLen}次。`); console.log(`此時序列為:${arr.join('')}\n---`); // 步驟3(關鍵步驟):把牌堆頂三張放到中間任意位置 arr = moveCardMiddleRandom(3, arr); console.log(`步驟3:把牌堆頂3張放到中間的隨機位置。`); console.log(`此時序列為:${arr.join('')}\n---`); // 步驟4(關鍵步驟):把最頂上的牌拿走 const restCard = arr.shift(); // 彈出隊頭元素 console.log(`步驟4:把最頂上的牌拿走,放在一邊。`); console.log(`拿走的牌為:${restCard}`); console.log(`此時序列為:${arr.join('')}\n---`); // 步驟5(無關步驟):根據(jù)南方人/北方人/不確定,把頂上的1/2/3張牌插入到中間任意位置 // 隨機選擇1、2、3中的任意一個數(shù)字 const moveNum = Math.floor(Math.random() * 3) + 1; arr = moveCardMiddleRandom(moveNum, arr); console.log(`步驟5:我${moveNum === 1 ? '是南方人' : moveNum === 2 ? '是北方人' : '不確定自己是哪里人'},\ 把${moveNum}張牌插入到中間的隨機位置。`); console.log(`此時序列為:${arr.join('')}\n---`); // 步驟6(關鍵步驟):根據(jù)性別男或女,移除牌堆頂?shù)?或2張牌 const maleNum = Math.floor(Math.random() * 2) + 1; // 隨機選擇1或2 for (let i = 0; i < maleNum; i++) { // 循環(huán)maleNum次,移除牌堆頂?shù)呐? arr.shift(); } console.log(`步驟6:我是${maleNum === 1 ? '男' : '女'}生,移除牌堆頂?shù)?{maleNum}張牌。`); console.log(`此時序列為:${arr.join('')}\n---`); // 步驟7(關鍵步驟):把頂部的牌移動到末尾,執(zhí)行7次 arr = moveCardBack(7, arr); console.log(`步驟7:把頂部的牌移動到末尾,執(zhí)行7次`); console.log(`此時序列為:${arr.join('')}\n---`); // 步驟8(關鍵步驟):執(zhí)行約瑟夫環(huán)過程。把牌堆頂一張牌放到末尾,再移除一張牌,直到只剩下一張牌。 console.log(`步驟8:把牌堆頂一張牌放到末尾,再移除一張牌,直到只剩下一張牌。`); while (arr.length > 1) { const luck = arr.shift(); // 好運留下來 arr.push(luck); console.log(`好運留下來:${luck}\t\t此時序列為:${arr.join('')}`); const sadness = arr.shift(); // 煩惱都丟掉 console.log(`煩惱都丟掉:${sadness}\t\t此時序列為:${arr.join('')}`); } console.log(`---\n最終結果:剩下的牌為${arr[0]},步驟4中留下來的牌也是${restCard}`);
這段代碼實現(xiàn)了昨晚春晚上劉謙的第二個魔術表演的過程,并提供了詳細的解釋。享受魔術的魅力吧!
以上就是JavaScript代碼實現(xiàn)春晚劉謙魔術的模擬程序的詳細內容,更多關于JavaScript劉謙魔術模擬程序的資料請關注腳本之家其它相關文章!
相關文章
JavaScript子類用Object.getPrototypeOf去調用父類方法解析
這篇文章主要介紹了JavaScript子類用Object.getPrototypeOf去調用父類方法。需要的朋友可以過來參考下,希望對大家有所幫助2013-12-12JavaScript子窗口調用父窗口變量和函數(shù)的方法
這篇文章主要介紹了JavaScript子窗口調用父窗口變量和函數(shù)的方法,涉及JavaScript窗口調用的相關技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-10-10JavaScript 正則應用詳解【模式、欲查、反向引用等】
這篇文章主要介紹了JavaScript 正則應用,結合實例形式詳細分析了JavaScript 正則表達式模式、欲查、反向引用等相關概念、原理與操作注意事項,需要的朋友可以參考下2020-05-05