Android面试-算法学习

排序

选择排序

  • 主要思想 在未排序序列中找出最小的元素,添加到有序序列中。
选择排序图解
  • 思路:
  1. 将整个序列分为无序区和有序区,初始时候有序区为空,无序区含有待排序的所有记录。
  2. 在无序区中选取最小的元素,将它与无序区的第一个元素交换,使得有序区扩展了一位,同时无序区减少了一位。
  3. 重复 2 ,直到无序区只剩下一个记录为之。此时所有的记录已经从小到大排序完毕。
  • 代码实现
    public static  void sort(int[] array){
        int len = array.length;
        for (int i = 0; i < len; i++){
            //对n个元素进行n-1次简单排序
            int index = i; 
            for(int j = i + 1; j < len; j++){
                //在这层循环里面找出最小的元素
                if(array[j] < array[index]){
                    index = j;
                }
            }
            if(index != i){
                int tem = array[i];
                array[i] = array[index];
                array[index] = tem;
            }
        }
    }

冒泡排序(稳定)

冒泡排序,又称起泡排序。是交换排序中最简单的排序方法。

  • 主要思想: 两两比较相邻的元素,如果反序则交换,直到没有反序。(反序就是违反序列规律,比如你要从小到大排序,但是两个元素 [5,2] 这样就反序了)

    起泡排序图解.jpg
  • 思路:

  1. 将整个序列分为无序区和有序区,初始时候有序区为空,无序区含有待排序的所有记录。
  2. 对无序区从前到后依次相邻的元素两两比较,反序则交换,从而使较小的元素往前移,较大的元素往后移,,一趟下来无序区中最大的元素就会在无序区的最后了,我们把这个元素移除无序区放到有序区中。(像水中的起泡,体积大的先浮上来)
  3. 重复 2 ,直到无序区只剩下一个记录为之。此时所有的记录已经从小到大排序完毕。
  • 再举个例子吧

    初始元素序列 [50 , 13 , 55 , 97 , 27, 38, 49, 65] 方括号代表无序区
    第一趟排序结果 [13 , 50 , 55 , 27 , 38 , 49 , 65 ], 97

    先是 50 和 13 比较,50>13,反序了,交换位置。

    然后 50 和 55 比较,50<55,正常,下一个。

    然后是 55 和 97 比较,55<97,正常,下一个。

    然后是 97 和 27 比较,97>27,反序,交换位置。

    ···

    如此类推,就把最大的元素97给放到最后了,此时97被排除出无序区,放进了有序区。大家可以把剩下几趟的排序结果自己排一下练手看看是否理解。

  • 代码实现

    public void sort(int[] array){
        //第一层循环从数组最后往前遍历
        for(int i = array.length - 1; i > 0; i--){
            //这里循环上界是i-1,体现出每趟排序选出的最大值从无序区中转移到有序区中
            for(int j = 0; j < i; j++){
                if(array[j] > array[j+1]){
                    int tem = array[j];
                    array[j] = array[j+1];
                    array[j+1] = tem;
                }
            }
        }
    }
    

快速排序(不稳定)

快速是对冒泡排序的一种改进。在冒泡排序中,记录的比较和移动是两两相邻的元素,把较大的元素一步一步推到后面,因此总的比较次数和移动次数较多。而快速排序是从两端向中间进行的。较大的元素一下子就能从前面移动到后面,这样比较的次数和移动的次数就减少了。

  • 主要思想:运用分治算法,找一个分割值key,然后将无序区元素分为小于 key 和大于 key的元素集,直到所有的分区只包含一个元素。

    快速排序图解.jpg

  • 思路:

    1. 选择分割值:最简单的方法是选择第一个元素作为分割值。
    2. 定义i、j 分别指向集合的最左和最右。
    3. 从右侧扫描小于key的,r[ i ] 和 r[ j ] 进行交换,并执行i++。
    4. 然后从左侧 i 开始扫描,找到大于key 的,r[ i ] 和 r[ j ] 进行交换,并执行 j-- 。
    5. 重复以上步骤,直到 i = j ;此时一次划分完毕,无序元素集合将分为小于key和大于key的两个无序区。
    6. 退出循环,说明 i 和 j 指向 key 所在的位置,返回该位置
  • 代码实现:

    //一次迭代
    public int partition(int[] r, int start, int end){
        int i = start;
        int j = end;
        while(i < j){
          //右侧扫描
          while(i < j && r[i] <= r[j]){
              j--;//一直到r[i] > r[j] ,代表找到了小于key的值,可以进行交换了
          }
          if(i < j){
              swap(r, i, j);//交换数组两个元素,此方法省略
              i++;
          }
          //然后是从左侧扫描
          while(i < j && r[i] <= r[j]){
              i++;
          }
          if(i < j){
              swap(r, i, j);
              j--;
          }
        }
      return i;
    }
    
    //递归实现
    public void queickSort(int[] r, int start, int end){
      if(end - start > 1){//此时一个区间长度大于1,递归才继续执行,否则结束
          int mid = 0;
          mid = partition(r, start, end);
          queickSort(r, start, mid);//对左侧无序区进行快速排序,分治处理
          queickSort(r, mid+1, end);//对右侧无序区进行快速排序,分治处理
      }
    }
    

归并排序 (稳定)

归并排序的中心思想还是分治。“归并”的含义就是将两个或者两个以上的有序序列归并成一个有序序列的过程。二路归并排序,是归并排序中最简单的排序方法。

归并排序.jpg

归并排序有个主要问题:如何将两个相邻的有序序列合并成一个有序序列?

因为两个有序序列,在归并过程中,可能会破坏掉原来的序列。像上图中,[20, 60],[10, 50] 在合并中就会破坏原来的序列了。为此,我们设置i, j, k 三个参数,分别指向两个待归并的序列,以及最终序列的当前位置。也就是 [ i→20 , j→10] ,k指向归并结果存放的位置,一开始是k→20 起始位置。

下面是一次归并的代码

/**
* r[start]→r[mid] , r[mid+1]→r[end]
* 下列start、mid、end 三个参数分别对应有序序列的位置如上
*/
void merge (int[] r, int[] merge, int start, int mid, int end){
      int i = start;
      int j = mid + 1;
      int k = start;
      while(i <= mid && j <= end){
          merge[k++] = r[i] >= r[j] ? r[j++] : r[i++];//将小的元素放入merge里面 
      }
      
      if(i <= mid){//第一个子序列没处理完
          while(i <= mid){
              merge[k++] = r[i++];
          }
      } else {
          while(j <= end){//第二个序列没处理完
              merge[k++] = r[j++];
          }
      }
      
      for (int index = start; index <= end; index++)
          r[index] = merge[index];
}

另外要注意的是,使用递归的方式,是先把原来的数组拆分成一个一个的元素,这时候 start>=end ,然后才会开始进行两两归并排序。

//递归实现
void mergeSort(int[] r, int[] merge, int start, int end){
      if(start >= end) return;
  
      int mid = (start + end)/2;
      mergeSort(r, merge, start, mid);
      mergeSort(r, merge, mid+1, end);
  
      merge(r, merge, start, mid, end);
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,233评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,013评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,030评论 0 241
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,827评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,221评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,542评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,814评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,513评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,225评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,497评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,998评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,342评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,986评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,055评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,812评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,560评论 2 271
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,461评论 2 266

推荐阅读更多精彩内容

  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    蚁前阅读 5,101评论 0 52
  • 概述:排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    每天刷两次牙阅读 3,706评论 0 15
  • 总结一下常见的排序算法。 排序分内排序和外排序。内排序:指在排序期间数据对象全部存放在内存的排序。外排序:指在排序...
    jiangliang阅读 1,268评论 0 1
  • 1.插入排序—直接插入排序(Straight Insertion Sort) 基本思想: 将一个记录插入到已排序好...
    依依玖玥阅读 1,177评论 0 2
  • 概述排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的...
    Luc_阅读 2,210评论 0 35