[C#][算法] 用菜鳥的思惟學習算法 -- 馬桶排序、冒泡排序和快速排序

用菜鳥的思惟學習算法 -- 馬桶排序、冒泡排序和快速排序

【博主】反骨仔    【來源】http://www.cnblogs.com/liqingwen/p/4994261.html   html

目錄

  • 馬桶排序(使人做嘔的排序)
  • 冒泡排序(面試都要問的算法)
  • 快速排序(見證亞當和夏娃的愛情之旅)

 

馬桶排序(使人做嘔的排序)

  1、場景:期末考試完了,老師要將同窗們的分數從高到低排序。假設班上有 5 名同窗,分別考了 5 分、3 分、5 分、2 分和 8 分【滿分:10 分】,排序後的結果就是 8 5 5 3 2,如今,讓咱們先思考 10 分鐘吧!
 
  2、思路
    (1)先建立一個數組 int scores[11],就有 scores[0]~scores[10] 共 11 個變量。咱們用變量值爲 0 表示沒有人獲得該分數,即 scores [0]=0 表示沒有人得 0 分,scores [10]=0 表示沒有人得 10 分,而 scores [8]=1 表示有一我的獲得 8 分。
 
    
    (2)第 1 個數爲 5,因此在 scores[5]=0 的基礎上+1,即 scores[5]=1 表示有 1 人獲得 5 分
 
    
    (3)第 2 個數爲 3,因此在 scores[3]=0 的基礎上+1,即 scores[3]=1 表示有 1 人獲得 3 分
 
    
    (4)第 3 個數爲 5,因此在 scores[5]=1 的基礎上+1,即 scores[5]=2 表示有 2 人獲得 5 分
 
         
      ... ...
    (5)依此類推,處理第 4 和第 5 個數,最終的結果圖以下:
 
 
    
    (6)咱們發現,scores[0]~scores[10] 內對應的值就是 0~10 分中每一個分數所出現的次數。如今,只需將結果打印便可,出現幾回就打印機次。
 
 
     咱們暫且稱它爲「馬桶排序」,這個算法就至關於有 11 個馬桶,編號從 0~10。每出現一個數,就在對應編號的馬桶中放一個旗子。
 
圖:這裏有 11 個馬桶
 
 
   3、思考:如今分別有 5 我的的名字和分數:小A 五、小二 三、小三 五、小妞 2 和王大錘 8,請按照分數從高到低,輸出他們的名字?
 
       【特色】
         假設須要排序的範圍 0~20000000,則須要 new int[20000001],很是浪費空間,即使只給 2 個數排序(1,19999999 );
         若是排序的數是小數不行,如:3.141 5926 5358 9793 2384 6264 3383 2795 0238;
 
  這裏使用 C# 給出簡單的算法過程。
 1         static void Main(string[] args)
 2         {
 3             var scores = new int[] { 5, 3, 5, 2, 8 };
 4             var newScores = new int[9];
 5 
 6             for (int i = 0; i < scores.Length; i++)
 7             {
 8                 var score = scores[i];
 9                 newScores[score]++;
10             }
11 
12             for (int i = newScores.Length - 1; i >= 0; i--)
13             {
14                 var num = newScores[i];
15                 for (int j = 1; j <= num; j++)
16                 {
17                     Console.Write($"{i} ");
18                 }
19             }
20 
21             Console.WriteLine();
22             Console.Read();
23         }
24     }    

 

冒泡排序(面試都要問的算法)

  1、基本思想:每次比較相鄰的兩個 元素,按需調整順序
 
  2、題目:要求將 12 35 99 18 76 這 5 個數進行從大到小排序
 
  3、思路
    (1)先比較第 1 位和第 2 位的大小,12<35,由於但願越小越靠後,因此要調整二者順序,交換後的結果:35 12 99 18 76
    (2)如今比較第 2 位和第 3 位的大小,12<99,因此須要交換位置,交換後的結果爲:35 99 12 18 76
    (3)接着比較第 3 位和第 4 位的大小,12<18,交換後的結果爲:35 99 18 12 76
    (4)最後比較第 4 位和第 5 位的大小,12<76,交換後的結果爲:35 99 18 76 12
 
 
    
    (5)通過 4 次後咱們發現 5 個數中最小的一個數已經就位,每將一個數歸位咱們稱其爲「一趟」;
    (6)如今咱們開始第二趟,目標將第 2 小的數歸位,根據以前邏輯,仍是從第 1 個數和第 2 個數開始比較上:
            35 99 18 76 12 --①--> 99 35 18 76 12 --②--> 99 35 18 76 12 --③--> 99 35 76 18 12
            在第一趟比較就知道第 5 位是最小的,因此第 4 位不用和第 5 位比較,這一趟只需比較 3 次
    (7)第3趟:99 35 76 18 12 --> 99 35 76 18 12 --> 99 76 35 18 12 (比較 2 次
    (8)第4趟:99 76 35 18 12 --> 99 76 35 18 12 ,有4個數已經就位,那麼最後一個數無須比較,它就是最大的
 
  【總結】若是有 n 個數進行排序,只需將 n-1 個數歸位,即要進行 n-1 趟操做,而每一趟開始都從第 1 位進行相鄰的兩個數 進行比較,將小的那個數放在後面,已經歸位的就不用進行比較。
  
  【特色】冒泡算法的核心部分是雙重嵌套循環,能夠看出時間複雜度是 O(N²),這是一個很是高的時間複雜度。
 
  這裏使用 C# 給出簡單的算法過程。 
 1         static void Main(string[] args)
 2         {
 3             var nums = new int[] { 12, 35, 99, 18, 76 };
 4             Output(nums);
 5 
 6             for (int j = 0; j < nums.Length - 1; j++)
 7             {
 8                 for (int i = 0; i < nums.Length - 1; i++)
 9                 {
10                     if (nums[i] < nums[i + 1])
11                     {
12                         var temp = nums[i];
13                         nums[i] = nums[i + 1];
14                         nums[i + 1] = temp;
15                     }
16                 }
17 
18                 Output(nums);
19             }
20 
21             Console.Read();
22         }
23 
24         /// <summary>
25         /// 控制檯輸出
26         /// </summary>
27         /// <param name="nums"></param>
28         static void Output(int[] nums)
29         {
30             foreach (var num in nums)
31             {
32                 Console.Write($"{num} ");
33             }
34 
35             Console.WriteLine();
36         }

 

快速排序(見證亞當和夏娃的愛情之旅)

  1、場景:對 6 1 2 7 9 3 4 5 10 8 這 10 個數進行排序
 
  2、思路
    先找一個基準數(一個用來參照的數),爲了方便,咱們選最左邊的 6,但願將 >6 的放到 6 的右邊,<6 的放到 6 左邊。如:3 1 2 5 4 6 9 7 10 8
    先假設須要將 6 挪到的位置爲 k,k 左邊的數 <6,右邊的數 >6
    (1)咱們先從初始數列「6 1 2 7 9 3 4 5 10 8 」的兩端開始「探測 」,先從右邊往左找一個 <6 的數,再從左往右找一個 >6 的數,而後交換。咱們用變量 i 和變量 j 指向序列的最左邊和最右邊。剛開始時最左邊 i=0 指向 6,最右邊 j=9 指向 8
 
 
 
圖:亞當 i 和夏娃 j 
 
    (2)如今設置的基準數是最左邊的數,因此序列先右往左移動(j--),當找到一個 <6 的數(5)就停下來。接着序列從左往右移動(i++),直到找到一個 >6 的數又停下來(7);
    (3)二者交換,結果:6 1 2 5 9 3 4 7 10 8;
 
 
 
    (4)j 的位置繼續向左移動(友情提示:每次都必須先從 j 的位置出發),發現 4 知足要求,接着 i++ 發現 9 知足要求,交換後的結果:6 1 2 5 4 3 9 7 10 8;
 

 
    (5)目前 j 指向的值爲 9,i 指向的值爲 4,j-- 發現 3 符合要求,接着 i++ 發現 i=j,說明這一輪移動結束啦。如今將基準數 6 和 3 進行交換,結果:3 1 2 5 4 6 9 7 10 8; 如今 6 左邊的數都是 <6 的,而右邊的數都是 >6 的,但遊戲還沒結束

 

    

  圖:亞當和夏娃終於產生了交集
 
    (6)咱們將 6 左邊的數拿出來先:3 1 2 5 4,此次以 3 爲基準數進行調整,使得 3 左邊的數 <3,右邊的數 >3,根據以前的模擬,此次的結果:2 1 3 5 4
    (7)再將 2 1 摳出來從新整理,獲得的結果: 1 2
    (8)剩下右邊的序列:9 7 10 8 也是這樣來搞,最終的結果: 1 2 3 4 5 6 7 8 9 10 (具體看下圖)
 
 
 
 
    【總結】快速排序的每一輪處理其實就是將這一輪的基準數歸位,當全部的基準數歸位,排序就結束啦
 
 
 
 

  【參考】文章改編與插圖來源《啊哈!算法》面試

相關文章
相關標籤/搜索