排序论

排序:

1、插入排序

首先说说插入排序。都知道插入排序是什么吗?假如你和朋友在玩牌,牌扣在桌上,我们手上没有牌。然后,我们每次从桌子上拿走一张牌并将它插入手中正确的位置。为了找到牌的正确位置,我们需要把目前所有牌搜索一遍。

给出伪代码(从小到大):

int insertion-sort()
{
    for(j=2;j<=n;j++)
    {
        key=a[j];
        //插入j进入排序序列
        i=j-1;
        while(i>0 && a[i]>key)
        {
            a[i+1]=a[i];
            i--;
        }
        a[i+1]=key;
    }
}

  这里我们先不定义任何数据。其实已经能理解了:先把要插入的数存入key,然后搜索现在已有的每一个数,只要数比它大,就一直搜下去。找到比它小的数,就插进去。我们推断出了这个算法是正确的,如果从大到小排序,相信大家都知道怎么办了吧!把while循环稍作变动:a[i]>key 改成a[i]<key即可。

掺和一下时间复杂度:

同冒泡排序时间复杂度

2、冒泡排序(同等于sort函数)

冒牌排序就是最简单、但搜索次数极多的“暴力搜索”。冒泡排序只和不停地找其它数据和自己进行比较然后互换,适合在需要排序的数据比较少的时候使用这种排序。当然,排序量过大,时间超限不是问题。这伪代码还用放吗?应该都知道吧?我们来看看时间复杂度:

这个时间复杂度最坏的情况下有个争议。有人说O(n^2),有人说O[n*(n-1)/2]。这里我要强调:时间复杂度应该是O[n*(n-1)/2],前者复杂度一般程序效率多余或程序有误。最好情况O(n)。

3、哈希排序(桶排序)

哈希排序适用于所有需要排序的数据比较小的时候使用这种排序。只要知道要排序的数据的大小,就可以定一个数组(桶),来存所有数据。这里我们开始打比方。

现在你有10个桶,你需要排序的一个最大数据是10。你走到第一个桶前,然后把所有手中的1都放进去。然后走到第二个桶,把所有手中的2都放进去……然后走到第10个桶,把所有手中的10都放进去。你手中已经没有需要排序的数据了。现在你回到第一个桶,从第一个通道第十个桶,开始依次报出每个桶里的数据。就这样,我们通过“桶”完成了排序。

来看看时间复杂度:

桶排序无论什么情况下时间复杂度都是O(m+n),m代表所有个桶里的数据数量。效率是不是非常高?可惜,这种算法空间效率特别低。如果只有10个数排序,而每个数高达1亿,那么你就得开个下标1亿的数组!占几十万甚至几百万KB很有可能!所以我们要因题而异,选择算法。

4、快速排序(同等于quicksort函数)

这个方法不牺牲空间,时间效率可以根据情况而异,而且时间复杂度普遍比冒泡排序小。我们还是要比方比方啦!

现在一串数上有两个士兵,分别从数据两头出发。首先最右边的哨兵往左走,此时最左边的哨兵站在一个数上,

时间: 2024-10-24 20:34:34

排序论的相关文章

算法学习——单链表快排

/**  * 以p为轴对start-end间的节点进行快排(包括start && 不包括end):  * 思路:  * 1.将头节点作为轴节点start,从start.next开始遍历,如果节点小于轴start的值,将该节点插入到轴节点后面:  * 2.将轴节点插入合适位置,即找到最后一个小于轴的节点,将该节点与轴节点值互换,此时就链表分为两部分,小于轴节点和大于轴节点:  * 3.递归的遍历2中两部分节点.  *   * @param p  * @param start  * @para

BZOJ 4517: [Sdoi2016]排列计数 错排+逆元

4517: [Sdoi2016]排列计数 Description 求有多少种长度为 n 的序列 A,满足以下条件: 1 ~ n 这 n 个数在序列中各出现了一次 若第 i 个数 A[i] 的值为 i,则称 i 是稳定的.序列恰好有 m 个数是稳定的 满足条件的序列可能很多,序列数对 10^9+7 取模. Input 第一行一个数 T,表示有 T 组数据. 接下来 T 行,每行两个整数 n.m. T=500000,n≤1000000,m≤1000000 Output 输出 T 行,每行一个数,表示

LeetCode:Rank Scores - 按分数排名次

1.题目名称 Rank Scores(按分数排名次) 2.题目地址 https://leetcode.com/problems/rank-scores/ 3.题目内容 按分数排名次,如果两个Id的分数一样,那么他们的名次是一样的,排名从1开始.注意,每组分数的名次,都是上一组分数名次加一. 例如,有这样一组数据: +----+-------+ | Id | Score | +----+-------+ | 1  | 3.50  | | 2  | 3.65  | | 3  | 4.00  | | 

(一一一)图文混排基础 -利用正则分割和拼接属性字符串

很多时候需要用到图文混排,例如聊天气泡中的表情,空间.微博中的表情,例如下图: 红心和文字在一起. 比较复杂的情况是表情夹杂在文字之间. 要实现这种功能,首先要介绍iOS中用于显示属性文字的类. 用于文字显示的类除了text属性之外,还有attributedText属性,这个属性是NSAttributedString类型,通过这个属性可以实现不同文字的不同字体.颜色甚至把图片作为文字显示的功能. 下面介绍这个字符串的使用. 以一条微博内容为应用场景,介绍如何从中找出表情.话题等内容,其中表情替换

目前流行的几种排课算法的介绍

通用高校排课算法研究----2 .目前流行的几种排课算法的介绍 2   目前流行的几种排课算法的介绍 2.1. 自动排课算法 1 .问题的描述 我们讨论的自动排课问题的简化描述如下: 设要安排的课程为{ C1 , C2 , ., Cn} ,课程总数为n , 而各门课程每周安排次数(每次为连续的2 学时) 为{ N1 , N2 , ., Nn} ;每周教学日共5 天,即星期一- 星期五;每个教学日最多安排4次课程教学,即1 - 2 节.3 - 4 节.5 - 6 节和7 - 8 节(以下分别称第1

ytu 1067: 顺序排号(约瑟夫环)

1067: 顺序排号Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 31  Solved: 16[Submit][Status][Web Board] Description 有n人围成一圈,顺序排号.从第1个人开始报数(从1到3报数),凡报到3的人退出圈子,问最后留下的是原来的第几号的那位. Input 初始人数n Output 最后一人的初始编号 Sample Input 3 Sample Output 2 HINT Source freepro

Essential WPF日历排程控件Essential Schedule for WPF

Essential Schedule for WPF是一款具有OutLook外观的WPF平台下的日历日程控件,利用该控件可以进行创建和管理约会和任务安排等.支持拖拉任务活动.调整大小.导入任务安排和导出日程安排等. 具体功能: 支持以天.周.一周工作时间.月为单位查看日程安排 支持时间间隔,使用户可以查看一天中的几个时间段 使用导航条进行时间导航 支持直接添加.编辑.删除日程活动 数据绑定功能允许你绑定到任何IEnumerable集合 控件提供了 Office14Black, Office14B

HDU 2048 错排

错排递推公式: d(n) = (n-1)*(d[n-1]+d[n-2]): 证明:将第n个元素放到第k处,第k处的元素如果放到第n处,就是d(n-2),否则,先假设放到第n处,然后错排,就是d(n-1): 1 #include <bits/stdc++.h> 2 3 using namespace std; 4 5 double fac[22] = {1,1}; 6 double d[22] = {0,0,1,2}; 7 8 int main() 9 { 10 for(int i=1;i<

快排的递归和非递归

常用的快排都是用递归写的,因为比较简单,但是可以用栈来实现非递归的快排. 第一种是递归的快排 #include<stdio.h> #include <stdlib.h> #include <time.h> int quick(int a[],int i ,int j) {     int tmp=0,key,b=0;     int im,jm;     im=i;     jm=j;     key=a[i];     if(i>j)         retur