美文网首页
iOS 排序相关复习

iOS 排序相关复习

作者: ProgramDouglas | 来源:发表于2020-04-05 16:52 被阅读0次

1.冒泡排序

  • 重复的比较数组中相邻的两个元素。如果一个元素比另一个元素大(小),那么就交换这两个元素的位置。重复这一比较直至最后一个元素。这一比较会重复n-1趟,每一趟比较n-j次,j是已经排序好的元素个数。
  • 每一趟比较都能找出未排序元素中最大或者最小的那个数字。这就如同水泡从水底逐个飘到水面一样。
  • 平均时间复杂度 O(n^2),最差时间复杂度 O(n^2)
for(int i = 0; i < num-1; i++) {
    for(int j = 0; j < num - 1 - i; j++) {
        if(array[j] < array[j+1]) {
            int tmp = array[j];
            array[j] = array[j+1];
            array[j+1] = tmp;
        }
    }
}

2.选择排序

  • 从数组的第i个元素开始到第n个元素,从i+1到n寻找最小的元素。将找到的最小元素和第i位元素交换。
  • 平均时间复杂度 O(n^2),最差时间复杂度 O(n^2)
for(int i = 0; i < num-1; i++) {
    for(int j = i+1; j < num - 1; j++) {
        if(array[i] < array[j]) {
            int tmp = array[i];
            array[i] = array[j];
            array[j] = tmp;
        }
    }
}

3、插入排序

  • 取i作为元素,认为i-1之前都已经排好序,从后向前扫描
  • 如果已经排好序的序列中元素大于新元素,则将该元素往右移动一个位置,重复该步骤直到已排好序的元素小于或等于新元素。在当前位置插入新元素
  • 平均时间复杂度 O(n^2),最差时间复杂度 O(n^2)
for(int i = 1; i < num-1; i++) {
    int temp = array[i];
    for(int j = i-1; j >= 0 && temp < array[j]; j--) {
        array[j+1] = array[j];
        array[j]= temp;
    }
}

4.快速排序 (原理解析 https://blog.csdn.net/yzllz001/article/details/50982841)

  • 快速排序使用分治策略来把一个串行分为两个子串行。
  • 从数列中挑出一个元素,称为 “基准”(pivot)
  • 重新排序数列,所有元素比基准值小的摆放在基准前面,所有元素比基准值大的摆在基准的后面。在这个分区退出之后,该基准就处于数列的中间位置。这个称为分区(partition)操作。
  • 递归地(recursive)把小于基准值元素的子数列和大于基准值元素的子数列排序。
  • 在每次的迭代(iteration)中,它会“基准”元素摆到它最后的位置去。
  • 平均时间复杂度 O(nlogN),最差时间复杂度 O(n^2)
void quicksort(int left,int right) 
{ 
    int i,j,t,temp; 
    if (left>right) return; 
                                
    temp=a[left]; //temp中存的就是基准数 
    i=left; 
    j=right; 
    while(i!=j) 
    { 
       //顺序很重要,要先从右边开始找 
       while(a[j]>=temp && i<j) 
            j--; 

       //再找右边的 
       while(a[i]<=temp && i<j) 
            i++; 

       //交换两个数在数组中的位置 
       if (i<j) { 
            t=a[i]; a[i]=a[j]; a[j]=t; 
       } 
    } 
    //最终将基准数归位 
    a[left]=a[i]; 
    a[i]=temp; 
                             
    quicksort(left,i-1);//继续处理左边的,这里是一个递归的过程 
    quicksort(i+1,right);//继续处理右边的 ,这里是一个递归的过程 
} 

快排退化成O(N^2)

  • 这个答案还得看枢轴(pivot)的选择策略。在快速排序的早期版本中呢,最左面或者是最右面的那个元素被选为枢轴,那最坏的情况就会在下面的情况下发生啦:
    1)数组已经是正序(same order)排过序的。
    2)数组已经是倒序排过序的。
    3)所有的元素都相同(1、2的特殊情况)

    因为这些案例在用例中十分常见,所以这个问题可以通过要么选择一个随机的枢轴,或者选择一个分区中间的下标作为枢轴,或者(特别是对于相比更长的分区)选择分区的第一个、中间、最后一个元素的中值作为枢轴。有了这些修改,那快排的最差的情况就不那么容易出现了,但是如果输入的数组最大(或者最小元素)被选为枢轴,那最坏的情况就又来了。

5.堆排序 (https://www.jianshu.com/p/938789fde325)

  • 构造初始堆。将给定无序序列构造成一个大顶堆(一般升序采用大顶堆,降序采用小顶堆)
  • 将堆顶元素与末尾元素进行交换,使末尾元素最大。然后继续调整堆,再将堆顶元素与末尾元素交换,得到第二大元素。如此反复进行交换、重建、交换。
//给定父节点的索引,得到左子节点的索引
#define LeftChild(i) (2*(i)+1)
//元素下沉方法
void PercDown(int A[], int i, int N)
{
    //子节点的索引号
    int child;
    //存储当前父节点元素的临时变量
    //(注:每一个节点都可以看作是其子树的根节点)
    int tmp;

    for (tmp = A[i]; LeftChild(i)<N; i = child)
    {
        child = LeftChild(i);
        //挑选出左、右子节点中较大者
        if (child != N-1 && A[child+1]>A[child])
        {
            child++;
        }
        //比较当前父节点与较大子节点
        if (A[i]<A[child])
        {
            //交换当前父节点处的元素值与较大子节点的元素值
                        tmp= A[i];
            A[i] = A[child];
                        A[child] = tmp;
        }
        else
        {
            break;
        }
        
    }
}

//堆排序
void HeapSort(int A[], int N)
{
    int i;
    //步骤一:创建大根堆
    for (i  = N/2;  i>=0; i--)
    {
        PercDown(A, i, N);

    }
    //步骤二:调整大根堆
    for ( i = N-1; i > 0; i--)
    {
         //首尾交换
        Swap(&A[0], &A[i]);
        //元素下沉
        PercDown(A, 0, i);
    }
}

//交换两个元素的位置
void Swap(int *num1, int * num2)
{
    int tmp = *num1;
    *num1 = *num2;
    *num2 = tmp;
}

//主函数
void main()
{
    int A[6] = {4,5,3,2,6,1};
    HeapSort(A, 6);
}

6.归并排序

  • 设数列长为N,将数列分开成小数列一共要logN步,每步都是一个合并有序数列的过程,时间复杂度可以记为O(N),故一共为O(N*logN)
//将有二个有序数列a[first...mid]和a[mid...last]合并。  
void mergearray(int a[], int first, int mid, int last, int temp[])  
{  
    int i = first, j = mid + 1;  
    int m = mid,   n = last;  
    int k = 0;  
      
    while (i <= m && j <= n)  
    {  
        if (a[i] <= a[j])  
            temp[k++] = a[i++];  
        else  
            temp[k++] = a[j++];  
    }  
      
    while (i <= m)  
        temp[k++] = a[i++];  
      
    while (j <= n)  
        temp[k++] = a[j++];  
      
    for (i = 0; i < k; i++)  
        a[first + i] = temp[i];  
}  

void mergesort(int a[], int first, int last, int temp[])  
{  
    if (first < last)  
    {  
        int mid = (first + last) / 2;  
        mergesort(a, first, mid, temp);    //左边有序  
        mergesort(a, mid + 1, last, temp); //右边有序  
        mergearray(a, first, mid, last, temp); //再将二个有序数列合并  
    }  
}  
  
bool MergeSort(int a[], int n)  
{  
    int *p = new int[n];  
    if (p == NULL)  
        return false;  
    mergesort(a, 0, n - 1, p);  
    delete[] p;  
    return true;  
}  

7.希尔排序

8.基数排序

常见问题:

100万以内的质数,怎么筛

欧拉筛法: 当一个数为素数的时候,它的倍数肯定不是素数。所以我们可以从2开始通过乘积筛掉所有的合数。将所有合数标记,保证不被重复筛除,时间复杂度为O(n)。
保证一个合数只被筛一次:任何的合数肯定有一个最小质因子。我们通过这个最小质因子就可以判断什么时候不用继续筛下去了。
当i是prime[j]的整数倍时(i % prime[j] == 0),i*prime[j+1]肯定被筛过,跳出循环

bool IsPrime[10000001];
int Pri[2000001],PriN;
int FindPrime ( int MaxN ) {
    for( int i = 2 ; i <= MaxN ; ++i ){
        if( IsPrime[ i ] )
            Pri[ PriN++ ]=i; //将这句话放在下面的循环前以保证PriN和Pri值的完整性
        for(int j=0;j<PriN;++j){
            if( i*Pri[ j ] > MaxN )
                break; //当过大了就跳出
            IsPrime[ i * Pri[ j ] ] = 0;
            //筛去素数
            if( i % Pri[ j ] == 0 ) break;
            //这里是关键,如果i是一个合数(这当然是允许的)而且i mod prime[j] = 0
            //那么跳出,因为i*prime[ (- over -)j ]一定已经被筛去了,被一个素因子比i小的数
        }
    }
}

堆排,快排,什么区别

在真实情况下, 一般快排的效率比堆排序高很多。
快排:数组中交换数据,在数据量不是特别大,而且离散程度较高的情况下效率很高
堆排序:创建堆,数据交换,调整堆的时间均很多
所以在实际应用中,我们用快排会更多一点。
堆排序的典型应用:在100万个数字中找出最大的100个这种问题,构建一个小顶堆然后遍历调整就可以了
为什么是小顶堆:小顶堆,最小的数就在最上面,只要与最上面的数进行比较就可以了,所以是小顶

topK问题 (维护K大小的最小堆)

https://blog.csdn.net/will130/article/details/49635429
https://www.cnblogs.com/xudong-bupt/archive/2013/03/20/2971262.html

相关文章

  • iOS 排序相关复习

    1.冒泡排序 重复的比较数组中相邻的两个元素。如果一个元素比另一个元素大(小),那么就交换这两个元素的位置。重复这...

  • iOS 排序相关

    根据对象里时间排序@"bg_updateTime"为时间字段,array为对象数组

  • 排序算法

    一、相关排序算法 1、IOS 经典排序算法总结:https://www.jianshu.com/p/ba57f02...

  • iOS 系统相关复习

    沙盒 iOS沙盒详细介绍iOS沙盒篇 沙盒机制介绍 iOS中的沙盒机制是一种安全体系。为了保证系统安全,iOS每个...

  • ios 相关知识复习

    1.CoreAnimation iOS动画篇_CoreAnimation(超详细解析核心动画) 扩展1彻底理解po...

  • iOS collectionView拖拽排序

    iOS collectionView拖拽排序 iOS collectionView拖拽排序

  • iOS算法总结-堆排序

    iOS算法总结-堆排序 iOS算法总结-堆排序

  • iOS算法总结-冒泡排序

    iOS算法总结-冒泡排序 iOS算法总结-冒泡排序

  • iOS Runloop

    这里记录下iOS中Runloop相关的知识点,以备以后复习总结。 先来说下Runloop相关的概念: Runloo...

  • Lucene--相关度排序和中文分析器

    一、相关度排序 1.什么是相关度排序 相关度排序是查询结果按照与查询关键字的相关性进行排序,越相关的越靠前。比如搜...

网友评论

      本文标题:iOS 排序相关复习

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