笔试时,冒泡排序也要写得优雅出众

作者: 三十六_ | 来源:发表于2016-06-01 16:13 被阅读5684次

导语

排序算法是笔试面试当中经常遇到的内容,之前参加的两个笔试都遇到了手写排序算法,可能大家都能写出来,但是要出众就要把代码优化一下,让人一看到就觉得你想的不简单。首先从最简单的冒泡排序开始,相信大家都能默出来,由于大部分人写的都一样,所以你要更出众的代码。

冒泡排序算法的步骤:
1.比较相邻的元素。如果第一个比第二个大,就交换他们两个。
2.对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对。最后的元素应该会是最大的数。
3.针对所有的元素重复以上的步骤,除了最后一个。
4.持续每次对越来越少的元素重复上面的步骤,直到没有任何一对数字需要比较。

首先给出严格按照定义写的代码:

void bubbleSort(int array[], int length)
{
    for (int i = 0; i < length; i++) {
        for (int j = 1; j < length - i; j++) {
            if (array[j-1] > array[j]) {
                swap(array[j-1], array[j]);
            }
        }
    }
}

这里先给出交换函数,交换函数有很多种写法,最基本的写法是:

void swap(int &a, int &b) {
    int temp = a;
    a = b;
    b = temp;
}

据说会在笔试中考不用中间变量交换两个数,网上有种写法:(其实这类方法还可已通过加减乘等代换方式实现,但都有一定弊端,虽然我们在平时的编程中不会这么写,不装X的话还是写第一种)

void swap(int &a, int &b) {
    a = a ^ b;
    b = b ^ a;
    a = a ^ b;
}

但这种写法有个问题,当交换同一元素时,该元素会被置零,在选择排序中会有bug所以还是建议第一种写法,当然你也可以在里面加个判断:

void swap(int &a, int &b) {
    if (a != b) {
        a = a ^ b;
        b = b ^ a;
        a = a ^ b;
    }
}

很明显上面的经典算法有些不足,当我们需要排序的数组基本有序时,上面的代码还会做出很多不必要的查找判断,降低了代码的执行效率。下面我们进行第一步优化,我们先定义一个标志flag,用来判断本次排序中是否发生交换,如果没有发生交换,说明排序已经完成,我们不需要再做不必要的循环判断,代码为:

void bubbleSort(int array[], int length) {
    bool flag = true;//判断是否发生交换
    while (flag) {
        flag = false;
        for (int j = 1; j < length; j++) {
            if (array[j-1] > array[j]) {
                swap(array[j-1], array[j]);
                flag = true;
            }
        }
        length --;
    }
}

对比下两种方法的比较大小次数:


第一种方法 第二种方法

由上图对比可见,第二种方法大大减少了对比的次数。

再做进一步优化,如果有数组前面几个是无序的,而后面的元素都已经是有序的,那我们就可以记录下无序的位置,下次排序判断时,只需要从数组头部遍历到该位置就可以了,这样可以省去遍历后面的元素,提高了代码的执行效率。代码为:

void bubbleSort(int array[], int length)
{
    int flag = length;
    while (flag > 0) {
        int k = flag;
        flag = 0;
        for (int j = 1; j < k; j++) {
            if (array[j - 1] > array[j]) {
                swap(array[j - 1], array[j]);
                flag = j;
            }
        }
    }
}

该方法和第二种的区别就是:先判断有没有交换,若有交换我也只遍历无序的区域。比较次数:

屏幕快照 2016-06-01 下午4.09.07.png

很明显比上面的少了很多次。
最后要说一句,冒泡排序写起来简单,相对于其他的排序算法效率比较低下,数据规模不大时,可以使用。

相关文章

  • 笔试时,冒泡排序也要写得优雅出众

    导语 排序算法是笔试面试当中经常遇到的内容,之前参加的两个笔试都遇到了手写排序算法,可能大家都能写出来,但是要出众...

  • 直接插入排序

    导语 接上一篇文章(冒泡排序也要写得优雅出众) http://www.jianshu.com/p/8abad5e9...

  • 常见的算法

    冒泡排序算法 冒泡排序作为一种最为常见的排序算法,很多企业的笔试当中经常出现,通常要求你手写冒泡排序,因此,...

  • 7.4-全栈Java笔记:三种经典算法

    冒泡排序算法 冒泡排序是最常用的排序算法,在笔试中也非常常见,能手写出冒泡排序算法可以说是基本的素养。 算法重复地...

  • PHP实现冒泡排序

    笔试时,常常遇到要手写实现PHP冒泡排序,虽说挺恶心的,但是还是得写出来

  • Java--冒泡排序算法

      冒泡排序是最常用的排序算法,在笔试中也非常常见,能手写出冒泡排序算法可以说是基本的素养。  算法重复地走访过要...

  • 写一个冒泡排序 二分查找、二分搜索

    用Java写一个冒泡排序。 答:冒泡排序几乎是个程序员都写得出来,但是面试的时候如何写一个逼格高的冒泡排序却不是每...

  • Java基础(冒泡排序与选择排序)

    冒泡排序 冒泡排序算法运行起来非常慢,但在概念上它是排序算法中最简单的,因此冒泡排序算法在刚开始研究排序技术时是一...

  • iOS程序员也要学点算法吧-简单排序之选择排序

    接之前的文章 iOS程序员也要学点算法吧-简单排序之冒泡排序 , 我们发现冒泡排序的比较效率和交换效率都是O(N²...

  • 问题[●●●]:谈谈你所知道的排序算法

    排序可以算是最基本的,最常用的算法,也是笔试面试中最常被考察到的算法。最基本的冒泡排序,选择排序,插入排序要可以很...

网友评论

  • 03eae4595aae:我感觉最后那里理论上还可以进一步优化,记录下第一个无序的位置pos,然后下次可以直接从pos-1的位置开始比较而不需要从数组头开始。当然记录这个第一次交换的位置的代价比较大!
  • e2030bd1a642:学到了
  • minghigh:把i看成1了…没错
  • minghigh:第一段代码边界是不是没考虑???
  • 洛朗不展傅立叶:啊,仔细想了一下,其实用异或运算做交换的时候不需要判断a和b是否相等啊,即使第一步异或之后a = 0,第二步b和0异或还是等于b,第三步,0和b异或等于b,a=b保持不变。可以不用判断哒,同样适用。
    三十六_:@洛朗不展傅立叶 注意这里所说的是同一元素,不是指值相同的元素,当我们在选择排序的时候可能会遇到array[0]就是最小的数,然后会调用swap(array[0], array[0]);这时它会被置零。
  • 洛朗不展傅立叶:啊,楼主棒棒哒,可以求github去follow吗?
  • 0be75bcce251:这个在JAVA里面有没有用 值传递值不会改变 两个数交换不了位置啊
    三十六_:@洛朗不展傅立叶 抱歉现在才回复你,&是取地址符,这里是用整型类型来保存变量地址,具体使用何种整型类型,取决于编译器的位数,你所说的这个方法其实是一样的,只不过调用时候传参不一样swap(&array[j-1], &array[j]);
    洛朗不展傅立叶:@_Coder int &a表示的是地址吗?可是这种变量是什么类型啊,可以作为函数参数吗?不是一般用指针变量:比如
    void swap(int *a_pointer, int *b_pointer)
    {
    int t = *a;
    *a = *b;
    *b = t;
    }
    这样吗?还是C语言出了什么新标准,我对CPP不了解,或者这个是CPP里的?看楼主好像运行出来了,不知道能不能看一下完整的代码orz
    三十六_: @59阿歪 这是C/C++的代码,java中没有地址引用符,这里swap的本质是通过传递变量地址(指针或引用)来交换地址中的值,java屏蔽了变量地址的概念,所以不能通过地址实现交换,但java可以使用其他方法,如利用数组交换
  • 扬仔360:无论走到多远多深,都不能忘了这种偏执的精神!
  • 389c20d5a244:很不错
  • T4Technology:感觉三种方法以前好像都见过。。。不过楼主总结的很好,又有图什么的,很详细。
    三十六_: @T4Technology 谢谢您的评论,排序算法有很多优化方法,我只是把我理解的用简单的语言写出来给大伙儿
  • 5552650dee15:看着不像是前端的写法
    三十六_: @z_小懒要奋斗 恩,不是前端语言,是C的
  • Orangels:第二个算法错了,length--应该在for循环外边
    三十六_:@Orangels 手误,差点对不起观众,已更正,十分感谢您的提醒,谢谢⊙_⊙
  • 兰州一碗面:笔试经常会看算法,楼主是学ios的妹子吗?
  • FrankLiu:楼主是一个善于思考的人,学习。
  • 清爽的丑八怪:人无我有,人有我优:+1:
  • 诸葛俊伟:挺有想法的
  • fe8246e7ae7f:如果一个人做事能注重这些细节,于细节处要求完美,他的专业能力应该不会差,至于面试还会成文问题吗
    4235b0b394bd:@左右君 术业有专攻,互相学习才有进步。
  • 4235b0b394bd:哈哈:smile:,太棒了,后天去面试太有用了。
  • 4fd0c8bd2f56:我是个小白,留着慢慢学
  • 492b9b7cf804:你很有想法
  • 34df88428320:这个笔试的时候还真的写过
  • 志城:赞\(≧▽≦)/

本文标题:笔试时,冒泡排序也要写得优雅出众

本文链接:https://www.haomeiwen.com/subject/kscblttx.html