亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

javascript隨機之洗牌算法深入分析

2019-11-20 14:45:09
字體:
來源:轉載
供稿:網友

洗牌算法是我們常見的隨機問題,在玩游戲、隨機排序時經常會碰到。它可以抽象成這樣:得到一個M以內的所有自然數的隨機順序數組。

在百度搜“洗牌算法”,第一個結果是《百度文庫-洗牌算法》,掃了一下里面的內容,很多內容都容易誤導別人走上歧途,包括最后用鏈表代替數組,也只是一個有限的優化(鏈表也引入了讀取效率的損失)。

該文里的第一種方法,可以簡單描述成:隨機抽牌,放在另一組;再次抽取,抽到空牌則重復抽。
“抽到空牌則重復抽”這會導致后面抽到空牌的機會越來越大,顯然是不合理的。
可以優化一步成:牌抽走后,原牌變少。(而不是留下空牌)
代碼如下:

復制代碼 代碼如下:

function shuffle_pick_1(m) //洗牌 //抽牌法
{
    //生成m張牌
    var arr = new Array(m);
    for (var i=0; i<m; i++) {
        arr[i] = i;
    }

    //每次抽出一張牌,放在另一堆。因為要在數組里抽出元素,把后面的所有元素向前拉一位,所以很耗時。
    var arr2 = new Array();
    for (var i=m; i>0; i--) {
        var rnd = Math.floor(Math.random()*i);
        arr2.push(arr[rnd]);
        arr.splice(rnd,1);
    }
    return arr2;
}

這個也明顯有問題,因為數組如果很大的話,刪除中間的某個元素,會導致后面的排隊向前走一步,這是一個很耗時的動作。
回想一下“我們為什么要刪除那個元素?”目的就是為了不產生空牌。
除了刪除那個元素之外,我們是不是還有其它方式來去除空牌?
----有的,我們把最后一張未抽的牌放在那個抽走的位置上就可以了。
所以,這個思路我們可以優化成這樣:

復制代碼 代碼如下:

function shuffle_pick(m) //洗牌 //抽牌法優化牌
{
    //生成m張牌
    var arr = new Array(m);
    for (var i=0; i<m; i++) {
        arr[i] = i;
    }

    //每次抽出一張牌,放在另一堆。把最后一張未抽的牌放在空位子上。
    var arr2 = new Array();
    for (var i=m; i>0;) {
        var rnd = Math.floor(Math.random()*i);
        arr2.push(arr[rnd]);
        arr[rnd] = arr[--i];
    }
    return arr2;
}


除了抽牌思路,我們還可以用換牌思路。
《百度文庫-洗牌算法》提到一種換牌思路:“隨機交換兩個位置,共交換n次,n越大,越接近隨機”。
這個做法是不對的,就算n很大(例如10張牌,進行10次調換),也還存在很大可能“有的牌根本沒換位置”。
順著這個思路,做一點小調整就可以了:第i張與任意一張牌換位子,換完一輪即可。
代碼如下:

復制代碼 代碼如下:

function shuffle_swap(m) //洗牌 //換牌法
{
    //生成m張牌
    var arr = new Array(m);
    for (var i=0; i<m; i++) {
        arr[i] = i;
    }

    //第i張與任意一張牌換位子,換完一輪即可
    for (var i=0; i<m; i++) {
        var rnd = Math.floor(Math.random()*(i+1)),
            temp = arr[rnd];
        arr[rnd] = arr[i];
        arr[i]=temp;
    }
    return arr;
}

除了抽牌與換牌的思路,我們還可以用插牌的思路:先有一張牌,第二張牌有兩個位置可隨機插入(第一張牌前,或后),第三張牌有三個位置可隨機插入(放在后面,或插在第一位,或插在第二位),依此類推
代碼如下:

復制代碼 代碼如下:

function shuffle_insert_1(m) //洗牌 //插牌法
{
    //每次生成一張最大的牌,插在隨機的某張牌前。因為要在數組里插入元素,把后面的所有元素向后擠一位,所以很耗時。
    var arr = [0];
    for (var i=1; i<m; i++) {
        arr.splice(Math.floor(Math.random()*(i+1)),0,i);
    }
    return arr;
}

以上的代碼也會有一些問題:就是隨著牌數的增多,插牌變得越來越困難,因為插牌會導致后面的很多牌都往后推一步。
當然,我們也可以適當的優化一下:先有n-1張牌,第n張牌放在最后,然后與任意一張牌互換位置。
代碼如下:

復制代碼 代碼如下:

function shuffle_insert(m) //洗牌 //插牌法優化版,可以用數學歸納法證明,這種洗牌是均勻的。
{
    //每次生成一張最大的牌,與隨機的某張牌換位子
    var arr = new Array(m);
    arr[0] = 0;
    for (var i=1; i<m; i++) {
        var rnd = Math.floor(Math.random()*(i+1));
        arr[i] = arr[rnd];
        arr[rnd] = i;
    }
    return arr;
}

好的,全部的代碼如下,有興趣的同學可以在自己的機器上試下,看下他們各自的執行效率、以及最后的結果是否是理論隨機。

 

復制代碼 代碼如下:

<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb2312">
<title>JK:javascript 洗牌算法 </title>

</head>
<body>
<script type="text/javascript">

function shuffle_pick_1(m) //洗牌 //抽牌法
{
    //生成m張牌
    var arr = new Array(m);
    for (var i=0; i<m; i++) {
        arr[i] = i;
    }

    //每次抽出一張牌,放在另一堆。因為要在數組里抽出元素,把后面的所有元素向前拉一位,所以很耗時。
    var arr2 = new Array();
    for (var i=m; i>0; i--) {
        var rnd = Math.floor(Math.random()*i);
        arr2.push(arr[rnd]);
        arr.splice(rnd,1);
    }
    return arr2;
}


function shuffle_pick(m) //洗牌 //抽牌法優化牌
{
    //生成m張牌
    var arr = new Array(m);
    for (var i=0; i<m; i++) {
        arr[i] = i;
    }

    //每次抽出一張牌,放在另一堆。把最后一張未抽的牌放在空位子上。
    var arr2 = new Array();
    for (var i=m; i>0;) {
        var rnd = Math.floor(Math.random()*i);
        arr2.push(arr[rnd]);
        arr[rnd] = arr[--i];
    }
    return arr2;
}


function shuffle_swap(m) //洗牌 //換牌法
{
    //生成m張牌
    var arr = new Array(m);
    for (var i=0; i<m; i++) {
        arr[i] = i;
    }

    //第i張與任意一張牌換位子,換完一輪即可
    for (var i=0; i<m; i++) {
        var rnd = Math.floor(Math.random()*(i+1)),
            temp = arr[rnd];
        arr[rnd] = arr[i];
        arr[i]=temp;
    }
    return arr;
}

function shuffle_insert_1(m) //洗牌 //插牌法
{
    //每次生成一張最大的牌,插在隨機的某張牌前。因為要在數組里插入元素,把后面的所有元素向后擠一位,所以很耗時。
    var arr = [0];
    for (var i=1; i<m; i++) {
        arr.splice(Math.floor(Math.random()*(i+1)),0,i);
    }
    return arr;
}

function shuffle_insert(m) //洗牌 //插牌法優化版,可以用數學歸納法證明,這種洗牌是均勻的。
{
    //每次生成一張最大的牌,與隨機的某張牌換位子
    var arr = new Array(m);
    arr[0] = 0;
    for (var i=1; i<m; i++) {
        var rnd = Math.floor(Math.random()*(i+1));
        arr[i] = arr[rnd];
        arr[rnd] = i;
    }
    return arr;
}


//alert(shuffle_pick(10))


var funcs = [shuffle_pick_1, shuffle_pick, shuffle_swap, shuffle_insert_1, shuffle_insert],
    funcNames = ["抽牌", "抽牌優化", "換牌", "插牌", "插牌優化"]
    m = 10000,
    times=[];
for(var i = 0; i < funcs.length; i++){
    var d0= new Date();
    funcs[i](m);
    funcNames[i] = (new Date() - d0) + '/t' + funcNames[i];
}

alert(funcNames.join('/n'));

</script>


</body>

</html>

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
成人免费大片黄在线播放| 日韩免费av片在线观看| 亚洲精品456在线播放狼人| 色偷偷亚洲男人天堂| 国产视频久久网| 一个色综合导航| 欧美久久精品午夜青青大伊人| 久久视频中文字幕| 国产精品极品美女在线观看免费| 亚洲在线视频观看| 在线成人免费网站| 国产欧美日韩中文字幕在线| 久久人91精品久久久久久不卡| 国产精品一二三视频| 97精品国产97久久久久久| 亚洲欧美在线x视频| 欧美日韩国产丝袜美女| 国内偷自视频区视频综合| 亚洲人成网站免费播放| 久久久久久久国产| 日韩电视剧免费观看网站| 亚洲男人天堂九九视频| 国产精品视频一区国模私拍| 在线视频亚洲欧美| 日韩电影中文字幕一区| 亚洲自拍偷拍福利| 亚洲国产精品资源| 日韩美女在线观看一区| 欧美日韩免费观看中文| 美女啪啪无遮挡免费久久网站| 国产视频精品va久久久久久| 日韩av不卡在线| 亚洲精品一区二区网址| 日本午夜在线亚洲.国产| 亚洲免费高清视频| 日韩激情av在线播放| 国产91色在线免费| yw.139尤物在线精品视频| 2021国产精品视频| 欧美电影免费观看大全| 国产精品av在线| 中文字幕在线日韩| 最新91在线视频| 欧美大成色www永久网站婷| 98精品在线视频| 国产日韩在线看片| 免费97视频在线精品国自产拍| 日本成人黄色片| 尤物九九久久国产精品的分类| 日韩欧美一区视频| www国产精品视频| 欧美成人合集magnet| 国产日韩换脸av一区在线观看| 欧美日韩亚洲天堂| 成人免费网站在线| 欧美精品在线第一页| 亚洲风情亚aⅴ在线发布| 日本精品久久电影| 亚洲一区二区三| 一区二区三区视频观看| 亚洲精品在线91| 欧美精品久久一区二区| 欧美日韩国产综合视频在线观看中文| 亚洲国产99精品国自产| 伊人久久久久久久久久久| 色噜噜亚洲精品中文字幕| 日韩中文字幕视频在线| 欧美巨大黑人极品精男| 亚洲第一网站免费视频| 国产亚洲精品久久久| 国产精品日韩欧美| 国产精品专区第二| 欧美日韩美女视频| 欧美高清不卡在线| 国产激情999| 亚洲日本aⅴ片在线观看香蕉| 欧美高清理论片| 国产精品第1页| 啪一啪鲁一鲁2019在线视频| 国产精品久久久久久久久久久久| 77777少妇光屁股久久一区| 欧美精品videossex88| 国产亚洲成av人片在线观看桃| 久久久久久伊人| 91福利视频网| 亚洲精品免费av| 91欧美精品午夜性色福利在线| 成人免费在线网址| 777午夜精品福利在线观看| 久久久久一本一区二区青青蜜月| 久久99久久99精品免观看粉嫩| 亚洲最新av网址| 久久九九国产精品怡红院| 国产日韩在线一区| 亚洲精品美女久久久| 亚洲精品久久视频| 青青久久av北条麻妃黑人| 日韩中文在线中文网三级| 国产一级揄自揄精品视频| 亚洲国产第一页| 亚洲国产福利在线| 欧美日韩国产中文精品字幕自在自线| 日韩女优在线播放| 91免费的视频在线播放| 91九色国产社区在线观看| 欧美另类69精品久久久久9999| 国产精品爱啪在线线免费观看| 欧美电影在线观看| 国产精品丝袜白浆摸在线| 亚洲午夜精品久久久久久久久久久久| 欧美国产日韩一区二区| 九九热这里只有精品免费看| 国产深夜精品福利| 精品亚洲永久免费精品| 日韩精品免费在线视频| 久久精视频免费在线久久完整在线看| 国产欧美一区二区三区久久人妖| 中文字幕亚洲欧美一区二区三区| 久久精品最新地址| 欧美俄罗斯性视频| 亚洲精品ady| 欧美视频在线免费看| 成人自拍性视频| 久久久久久久久久久免费| 亚洲欧美综合精品久久成人| 色一情一乱一区二区| 国产欧美在线播放| 欧美大学生性色视频| 欧美成人免费一级人片100| 国产一区二区黑人欧美xxxx| 国产精品免费视频xxxx| 亚洲精品日韩激情在线电影| 欧美做受高潮电影o| 在线丨暗呦小u女国产精品| 夜夜嗨av色综合久久久综合网| 欧美日韩国产丝袜美女| www.日本久久久久com.| 久久久精品美女| 日韩国产欧美精品一区二区三区| 91av视频在线播放| 亚洲国产成人91精品| 亚洲欧美日韩视频一区| 91精品久久久久久久久不口人| 亚洲一区二区久久久久久| 久久最新资源网| 精品国产91乱高清在线观看| 亚洲国产91精品在线观看| 国产精品入口免费视频一| 精品少妇v888av| 国产婷婷97碰碰久久人人蜜臀| 欧美一级淫片aaaaaaa视频| 亚州欧美日韩中文视频| www.精品av.com| 国产欧亚日韩视频| 国产剧情久久久久久| 国产美女精品视频| 日韩美女在线看| www.国产精品一二区| 欧美激情免费观看| 精品久久久久久久久久久久久| 亚洲国产精品va在线观看黑人| 亚洲免费av电影| 色婷婷av一区二区三区久久| 亚洲香蕉成人av网站在线观看|