亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

JavaScript代碼實現(xiàn)春晚劉謙魔術的模擬程序

 更新時間:2024年02月11日 08:13:31   作者:小u  
昨晚春晚上劉謙的兩個魔術表演都非常精彩,尤其是第二個魔術,他演繹了經典的約瑟夫環(huán)問題!約瑟夫環(huán)是一個經典的數(shù)學問題,本文給出了完整的 JavaScript 代碼實現(xiàn),感興趣的同學可以自己動手實現(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劉謙魔術模擬程序的資料請關注腳本之家其它相關文章!

相關文章

  • JS中的Select框實現(xiàn)模糊搜索功能

    JS中的Select框實現(xiàn)模糊搜索功能

    本文通過實例代碼介紹JS中的Select框實現(xiàn)模糊搜索功能,下面是使用JavaScript來實現(xiàn)模糊搜索的功能,代碼分為html部分和javascript部分,感興趣的朋友跟隨小編一起看看吧
    2024-06-06
  • 淺談頁面裝載js及性能分析方法

    淺談頁面裝載js及性能分析方法

    這篇文章主要簡單介紹了頁面裝載js及性能分析方法的相關資料,需要的朋友可以參考下
    2014-12-12
  • 如何利用Proxy更優(yōu)雅地處理異常詳解

    如何利用Proxy更優(yōu)雅地處理異常詳解

    這篇文章主要給大家介紹了關于如何利用Proxy更優(yōu)雅地處理異常的相關資料,文中通過實例代碼以及圖文介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2022-03-03
  • JavaScript子類用Object.getPrototypeOf去調用父類方法解析

    JavaScript子類用Object.getPrototypeOf去調用父類方法解析

    這篇文章主要介紹了JavaScript子類用Object.getPrototypeOf去調用父類方法。需要的朋友可以過來參考下,希望對大家有所幫助
    2013-12-12
  • JavaScript子窗口調用父窗口變量和函數(shù)的方法

    JavaScript子窗口調用父窗口變量和函數(shù)的方法

    這篇文章主要介紹了JavaScript子窗口調用父窗口變量和函數(shù)的方法,涉及JavaScript窗口調用的相關技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-10-10
  • JavaScript中內置函數(shù)Map()的使用

    JavaScript中內置函數(shù)Map()的使用

    Map()是JavaScript中內置的一種數(shù)據(jù)結構,它允許您將鍵值對映射到任意類型的值,主要介紹了JavaScript中內置函數(shù)Map()的使用,感興趣的可以了解一下
    2023-05-05
  • Typescript定義多個接口類型聲明的方式小結

    Typescript定義多個接口類型聲明的方式小結

    這篇文章主要介紹了Typescript定義多個接口類型聲明的方式小結,在 TypeScript 中,您可以使用交叉類型(&)或聯(lián)合類型(|)來組合多個接口,從而實現(xiàn)多個接口類型的混合,文中通過代碼講解的非常詳細,需要的朋友可以參考下
    2025-01-01
  • 淺談微信小程序列表埋點曝光指南

    淺談微信小程序列表埋點曝光指南

    這篇文章主要介紹了微信小程序列表埋點曝光指南,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-10-10
  • JavaScript 正則應用詳解【模式、欲查、反向引用等】

    JavaScript 正則應用詳解【模式、欲查、反向引用等】

    這篇文章主要介紹了JavaScript 正則應用,結合實例形式詳細分析了JavaScript 正則表達式模式、欲查、反向引用等相關概念、原理與操作注意事項,需要的朋友可以參考下
    2020-05-05
  • Bootstrap實現(xiàn)提示框和彈出框效果

    Bootstrap實現(xiàn)提示框和彈出框效果

    這篇文章主要為大家詳細介紹了Bootstrap實現(xiàn)彈出框和提示框效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-01-01

最新評論