算法作为程序猿的一种基本技能,对我来说却一直是一个硬伤。毕竟平时的工作接触算法的地方很少,大部分时间都是在搭建UI,写业务逻辑,写网络请求。然而在面试当中,算法往往又是考察的重点。

也许你会说这些平时工作用得到么?确实,用到的很少,但是你不会,人家就是不要你,你能怎样?!

关于这个事情,唐boy有篇文章,我觉得说的挺好,看完后也许你会所感触。http://blog.devtang.com/blog/2015/06/16/talk-about-tech-interview/

好了,废话不多说。关于排序算法,网上有很多资料,我从中整理了一下,并且写了详细的注释,为的是以后预习的时候能一目了然的看到每个算法的思想。如有不足,欢迎指正!

冒泡排序

/**
* 冒泡排序
* 对数组中每个位置的数据,从后往前推,依次比较相邻的两个数,如果后面的数较小,则交换两者位置
* 如果一次遍历没有发生任何数据交换,则排序直接完成
* @param array 数组
*/
+(void)bubbleSort:(NSMutableArray *)list{
if (list.count <= ) {
return;
}
int i, y;
BOOL bFinish = YES; //是否发生数据交换
for (i = ; i<= [list count] && bFinish; i++) {
bFinish = NO; //每次遍历时,重置标志
//从最后一位开始,依次跟前一位相比,如果较小,则交换位置
//当一次遍历没有任何数据交换时,则说明已经排序完成(bFinish=YES),则不再进行循环
for (y = (int)[list count]-; y>=i; y--) {
if ([[list objectAtIndex:y] intValue] < [[list objectAtIndex:y-] intValue]) {
//交换位置
// NSLog(@"%d<->%d",[[array objectAtIndex:y-1] intValue],[[array objectAtIndex:y] intValue]);
[list exchangeObjectAtIndex:y- withObjectAtIndex:y];
bFinish = YES; //发生数据交换,则继续进行下一次遍历,直到未发生数据交换或者循环完成为止 // NSLog(@"%@",[array componentsJoinedByString:@" "]);
} }
}
}
/**
* 冒泡排序(倒叙)
*
* @param list 待排序的数组
*/
+(void)bubbleSortDesc:(NSMutableArray *)list{
if (list.count <= ) {
return;
}
int i, y;
BOOL bFinish = YES; //是否发生数据交换
for (i = ; i<= [list count] && bFinish; i++) {
bFinish = NO; //每次遍历时,重置标志
//从最后一位开始,依次跟前一位相比,如果较小,则交换位置
//当一次遍历没有任何数据交换时,则说明已经排序完成(bFinish=YES),则不再进行循环
for (y = (int)[list count]-; y>=i; y--) {
if ([[list objectAtIndex:y] intValue] > [[list objectAtIndex:y-] intValue]) {
//交换位置
// NSLog(@"%d<->%d",[[array objectAtIndex:y-1] intValue],[[array objectAtIndex:y] intValue]);
[list exchangeObjectAtIndex:y- withObjectAtIndex:y];
bFinish = YES; //发生数据交换,则继续进行下一次遍历,直到未发生数据交换或者循环完成为止 // NSLog(@"%@",[array componentsJoinedByString:@" "]);
} }
}
}

快速排序

/**
* 快速排序
*
* @param list array
*/
+(void)quickSort:(NSMutableArray *)list{
if (list.count <= ) {
return;
}
[self quickSort:list startIndex: endIndex:list.count-];
} /**
* 快速排序
* 任意选取一个数据(通常选用数组的第一个数)作为关键数据,然后将所有比它小的数都放到它前面,所有比它大的数都放到它后面
* 再分别对两边的数据进行快速排序
* @param list 数组
* @param start 低位索引
* @param end 高位索引
*/
+(void)quickSort:(NSMutableArray *)list startIndex:(NSInteger)start endIndex:(NSInteger)end{
if (start >= end) { //低位大于高位,排序结束
return;
}
NSInteger low = start;
NSInteger high = end;
NSInteger key = [[list objectAtIndex:start] integerValue]; //取第一个数作为关键数据
while (low < high) { //从后往前推,直到找到第一个比关键数据小的值
while ([[list objectAtIndex:high] integerValue] >= key && low < high) {
high--;
}
//将这个值与关键数据对调(关键数据处于low位置),对调完关键数据处于high位置
[list exchangeObjectAtIndex:low withObjectAtIndex:high]; //从前往后推,直到找到第一个比关键数据大的值
while ([[list objectAtIndex:low] integerValue] <= key && low < high) {
low++;
}
//将这个值与关键数据(关键数据已经处于high位置)对调,对调完关键数据处于low位置
[list exchangeObjectAtIndex:high withObjectAtIndex:low];
}
//对关键数据前面的数据进行快速排序
[self quickSort:list startIndex:start endIndex:low-];
//对关键数据后面的数据进行快速排序
[self quickSort:list startIndex:low+ endIndex:end];
}

直接插入排序

/**
* 直接插入排序
* 把待排序的纪录按其关键码值的大小逐个插入到一个已经排好序的有序序列中,直到所有的纪录插入完为止
*
* @param list 待排序的数组
*/
+(void)insertionSort:(NSMutableArray *)list{
if (list.count <= ) {
return;
} //从第1位开始,依次将数组分成2部分:前一部分可以视为已经排好序的,后一部分是未排序的
//对后一部分的数据依次遍历,插入到前一部分中的合适位置
for (int i=; i<list.count; i++) {
int j=i;
//待排序的数(是未排序部分的第1个数,它的上一位数就是已经排序的部分的最后一位数)
NSInteger temp = [[list objectAtIndex:i] integerValue];
//从已排序部分的最后一位开始依次往前推,如果比待排序的数大,则将其位置往后移一位
while (j> && [[list objectAtIndex:(j-)]integerValue] > temp) {
[list replaceObjectAtIndex:j withObject:[list objectAtIndex:(j-)]];
j--;
}
//循环结束后的j即为待排序的数需要插入的位置
[list replaceObjectAtIndex:j withObject:@(temp)];
}
}

二分排序(插入排序)

/**
* 二分排序(插入排序)
*
* @param list 待排序的数组
*/
+(void)binaryInsertionSort:(NSMutableArray *)list{
if (list.count <= ) {
return;
}
//从第1位开始,依次将数组分成2部分:前一部分可以视为已经排好序的,后一部分是未排序的
//对后一部分的数据依次遍历,插入到前一部分中的合适位置
for (int i=; i<list.count; i++) {
int left = ;
int right = i-;
int middle;
//待排序的数(是未排序部分的第1个数,它的上一位数就是已经排序的部分的最后一位数)
NSInteger temp = [[list objectAtIndex:i] integerValue]; while (left <= right) {
//每次从已经排序的部分中取中间位置的数进行比较
middle = (left+right)/;
//如果跟待排序数相等,则直接插入到此位置即可
if ([[list objectAtIndex:middle]integerValue] == temp){
left = middle;
break;
}
//如果比待排序数大,则从中间位置左边范围内再次取中间数查找(下一循环)
else if ([[list objectAtIndex:middle]integerValue] > temp) {
right = middle-;
}
//如果比待排序数小,则从中间位置右边范围再次取中间数查找(下一循环)
else{
left = middle+;
}
}
//循环结束,找到待插入位置left
//依次将left右边(比left位置数据都大)的数据向右移动一位
for (int j=i; j>left; j--) {
[list replaceObjectAtIndex:j withObject:[list objectAtIndex:j-]];
}
//在left位置插入待排序数
[list replaceObjectAtIndex:left withObject:@(temp)];
}
}

希尔排序

/**
* 希尔排序(插入排序)
* 先取一个正整数d1<n,把所有序号相隔d1的数组元素放一组,组内进行直接插入排序;然后取d2<d1,重复上述分组和排序操作;直至di=1,即所有记录放进一个组中排序为止
*
* @param list 待排序的数组
*/
+(void)shellInsertionSort:(NSMutableArray *)list{
if (list.count <= ) {
return;
}
NSInteger delt = list.count / ; //增量:取数组长度一半,以后每次减半,直到增量为1
NSInteger i;
while (delt >= ) {
//对位置距离=增量的组分别进行直接插入排序
for (i=delt; i<list.count; i++) {
NSInteger k=i;
//待排序的数
NSInteger temp = [[list objectAtIndex:i] integerValue];
//从已排序部分的最后一位开始依次往前推(间隔=增量),如果比待排序的数大,则将其位置往后移“增量”位
while (k>=delt && [[list objectAtIndex:k-delt]integerValue] > temp) {
[list replaceObjectAtIndex:k withObject:[list objectAtIndex:k-delt]];
k -= delt; //按增量递推
}
//循环结束后的j即为待排序的数需要插入的位置
[list replaceObjectAtIndex:k withObject:@(temp)];
}
delt = delt/; //增量:每次减半
}
}

堆排序

/**
* 堆排序
* 利用堆的特性(堆顶肯定是最大或者最小值),将堆顶数据与堆尾数据交换(相当于删除堆顶数据并保存到堆尾),再重新构造堆,直到堆中只剩下1个数据
*
* @param list 待排序的数组
*/
+(void)heapSort:(NSMutableArray *)list{
if (list.count <= ) {
return;
}
NSInteger i;
//初始化堆,取父节点和左右子节点中最大(或者最小)值作为堆顶
//因为堆节点i的叶子节点是2*i+1,因此初始化是从length/2-1处开始整理堆即可
for(i=list.count/-;i>=;i--){
[self heapAdjust:list withIndex:i andLength:list.count];
}
// NSLog(@"构造堆:%@", [list componentsJoinedByString:@" "]);
//整理完堆后,第0位数肯定是最大(或者最小)值,将它跟第n-1位交换,相当于从堆中删除第0位,再对剩下的数据重新整理堆
//依次执行上述交换操作,直到只剩下1个数
for(i=list.count-;i>;i--){
[list exchangeObjectAtIndex: withObjectAtIndex:i];
[self heapAdjust:list withIndex: andLength:i];
// NSLog(@"构造堆:%@", [list componentsJoinedByString:@" "]);
}
}
/**
* 构造堆(最大堆)
*
*
* @param list 待整理的数组
* @param index 待整理的位置
* @param length 待整理的数组的长度(这里因为待排序的数据和已排序的数据共用了一个数组)
*/
+(void)heapAdjust:(NSMutableArray *)list withIndex:(NSInteger)index andLength:(NSInteger)length{
//左子节点,右子节点为lchild+1
NSInteger lchild = index*+; while (lchild < length) {
//判断是否有右子节点,如果有则判断右子节点是否比左节点大,如果比左节点大则用右子节点于父节点进行比较
if (lchild+ < length && [[list objectAtIndex:lchild+]integerValue] > [[list objectAtIndex:lchild] integerValue]) {
lchild++; //++表示将左节点位置移到右节点位置,为的是下面跟父节点进行比较
}
//如果父节点比子节点小,则说明本身就是一个堆,无需再整理
if ([[list objectAtIndex:lchild]integerValue] < [[list objectAtIndex:index]integerValue]) {
break;
}
//交换父节点和子节点的值(将父节点下沉)
[list exchangeObjectAtIndex:index withObjectAtIndex:lchild];
//父节点位置下沉到子节点的位置,再继续整理下面的堆
index = lchild;
lchild = index*+;
}
}

选择排序

/**
* 选择排序
* 每一次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,直到全部待排序的数据元素排完
*
* @param list 待排序的数组
*/
+(void)selectSort:(NSMutableArray *)list{
if (list.count <= ) {
return;
}
NSInteger i,j;
//从第0位开始排序,直到最后一位(最后一位无需再排序)
for (i=; i<list.count-; i++) {
j=i;//记录当前位置,作为对比的关键元素
//依次遍历后面的元素
for (NSInteger k=i+; k<list.count; k++) {
//如果元素比关键元素小,则将关键元素位置指向此位置,再继续遍历
if ([[list objectAtIndex:k]integerValue] < [[list objectAtIndex:j]integerValue]) {
j=k;
}
}
//遍历结束后,得到最小元素位置j,如果不是初始位置,则交换2个位置的值
if (i!=j) {
[list exchangeObjectAtIndex:i withObjectAtIndex:j];
}
}
}

归并排序

/**
* 归并排序
* 将数组分割成2部分,使左右2部分都有序,再合并成一个
* 为了使左右2部分都有序,可通过递归的方式,再对其进行分割,直至分割后只剩1个元素,可以视其为已经有序的,再将左右2部分进行合并
* 合并时,分别从2部分中取第0个数进行比较,将较小(或者较大)的数保存到临时数组里,再用下一个数和上次比较中较大者进行比较,直到某一边没有数据,再将另外一边中剩余的数复制到临时数组里
* 最后生成的临时数组就是已经排好序的结果
*
* @param list 待排序的数组
*/
+(void)mergeSort:(NSMutableArray *)list{
if (list.count <= ) {
return;
}
[self mergeSort:list startIndex: endIndex:list.count-];
}
+(void)mergeSort:(NSMutableArray *)list startIndex:(NSInteger)startIndex endIndex:(NSInteger)endIndex{ if (startIndex < endIndex) { //取中间位置,将左右两边分别递归进行归并,直至左右两边只剩1个元素
NSInteger middle = (startIndex + endIndex) / ;
[self mergeSort:list startIndex:startIndex endIndex:middle];
[self mergeSort:list startIndex:middle+ endIndex:endIndex];
//对左右2边的数据进行合并
NSInteger i=startIndex; //左边数据的起始位置
NSInteger j= middle+; //右边数据的起始位置
NSMutableArray *temp = [NSMutableArray array]; //临时数组
while (i<=middle && j <= endIndex) {
//如果左边数据较小,则将其放到临时数组里,并将左边位置向后移一位
if ([[list objectAtIndex:i]integerValue] < [[list objectAtIndex:j]integerValue]) {
[temp addObject:[list objectAtIndex:i]];
i++;
}
//否则将右边数据放到临时数组里,并将右边位置向后移一位
else{
[temp addObject:[list objectAtIndex:j]];
j++;
}
}
//如果左边还有数据,则将剩余的部分全都复制到临时数组里
while (i<=middle) {
[temp addObject:[list objectAtIndex:i]];
i++;
}
//如果右边还有数据(左右两边只可能存在一边有剩余数据的情况),则将剩余的部分全都复制到临时数组里
while (j<=endIndex) {
[temp addObject:[list objectAtIndex:j]];
j++;
}
//再将临时数组里的数据(已经排好序了)保存到原始数据中,以达到对原始数据的排序
for (i=; i<temp.count; i++) {
//注意:需要从startIndex位置开始,因为这里是递归调用的
[list replaceObjectAtIndex:startIndex withObject:[temp objectAtIndex:i]];
startIndex++;
}
}
}
 

Objective-C排序算法实现的更多相关文章

  1. JavaScript实现常用的排序算法

    ▓▓▓▓▓▓ 大致介绍 由于最近要考试复习,所以学习js的时间少了 -_-||,考试完还会继续的努力学习,这次用原生的JavaScript实现以前学习的常用的排序算法,有冒泡排序.快速排序.直接插入排 ...

  2. 排序算法----基数排序(RadixSort(L))单链表智能版本

    转载http://blog.csdn.net/Shayabean_/article/details/44885917博客 先说说基数排序的思想: 基数排序是非比较型的排序算法,其原理是将整数按位数切割 ...

  3. 常见排序算法(附java代码)

    常见排序算法与java实现 一.选择排序(SelectSort) 基本原理:对于给定的一组记录,经过第一轮比较后得到最小的记录,然后将该记录与第一个记录的位置进行交换:接着对不包括第一个记录以外的其他 ...

  4. 几大排序算法的Java实现

    很多的面试题都问到了排序算法,中间的算法和思想比较重要,这边我选择了5种常用排序算法并用Java进行了实现.自己写一个模板已防以后面试用到.大家可以看过算法之后,自己去实现一下. 1.冒泡排序:大数向 ...

  5. 排序算法----基数排序(RadixSort(L,max))单链表版本

    转载http://blog.csdn.net/Shayabean_/article/details/44885917博客 先说说基数排序的思想: 基数排序是非比较型的排序算法,其原理是将整数按位数切割 ...

  6. 排序算法汇总(C/C++实现)

    前言:     本人自接触算法近2年以来,在不断学习中越多地发觉各种算法中的美妙.之所以在这方面过多的投入,主要还是基于自身对高级程序设计的热爱,对数学的沉迷.回想一下,先后也曾参加过ACM大大小小的 ...

  7. 用Java来写常见的排序算法

    随着校招的临近 算法是校招中很重要的一个部分 总结了常见几种排序算法,各种算法的时间复杂度和空间复杂度大家也需要多了解下 package com.huwei.sort; /** * 各种排序算法 * ...

  8. 模板化的七种排序算法,适用于T* vector<T>以及list<T>

    最近在写一些数据结构以及算法相关的代码,比如常用排序算法以及具有启发能力的智能算法.为了能够让写下的代码下次还能够被复用,直接将代码编写成类模板成员函数的方式,之所以没有将这种方式改成更方便的函数模板 ...

  9. 排序算法总结第二弹----冒泡排序---javascript描述

    上篇博文总结了选择排序,这篇来看冒泡排序,接上篇. 冒泡排序思想:若是正再将一组数据升序排序, 第一趟:比较相邻的数据,当左侧值大于右侧值将他们进行交换,将较小值向前浮动,大值向后冒泡,直至比较到最后 ...

  10. 排序算法总结------选择排序 ---javascript描述

    每当面试时避不可少谈论的话题是排序算法,上次面试时被问到写排序算法,然后脑袋一懵不会写,狠狠的被面试官鄙视了一番,问我是不是第一次参加面试,怎么可以连排序算法都不会呢?不过当时确实是第一次去面试,以此 ...

随机推荐

  1. 日志记录到txt文件

    using System;using System.Collections.Generic;using System.IO;using System.Linq;using System.Web;usi ...

  2. Some tips on using HashSet<T> and List<T>

    This article is written based on my colleague's review Most of the times, when I want to use a colle ...

  3. Android输入输出机制之来龙去脉

    openInputChannelPair( 阅读本文的前提条件是知道匿名管道和匿名共享内存是怎么一回事,否则阅读相应的文章. Anonymous pipes 和Anonymous Shared Mem ...

  4. SQL Server 2008数据库重命名方法

    假设SQL Server 2008中有个数据库test,现在要将其改名为zhy步骤:(1) 分离数据库:打开management studio,找到test数据库-->右键-->任务--& ...

  5. GCC 编译选项(转)

    转:http://www.cnblogs.com/xmphoenix/archive/2011/03/21/1989944.html gcc提供了大量的警告选项,对代码中可能存在的问题提出警 告,通常 ...

  6. docker镜像的操作

    在主机上列出镜像 sudo docker images 每从Docker Hub下载一个镜像就会启动相对的创建一个容器 在镜像列表中看到三个重要的东西: 来自什么镜像源,例如ubuntu 每个镜像都有 ...

  7. ProgressBar的Indeterminate属性

    Indeterminate ProgressBar默认是白色的,如果容器的背景也是白色的,这样就根本看不到Progressbar. 简单解决方案: 用style属性设定反转的颜色. <Progr ...

  8. Cache的原理、设计及实现

    Cache的原理.设计及实现 前言 虽然CPU主频的提升会带动系统性能的改善,但系统性能的提高不仅仅取决于CPU,还与系统架构.指令结构.信息在各个部件之间的传送速度及存储部件的存取速度等因素有关,特 ...

  9. Java Properties类

    Properties 是哈希表的一个子类.它是用来维持值列表,其中的键是一个字符串,值也是一个字符串. Properties类被许多其他的Java类使用.例如,它是对象通过System.getProp ...

  10. Android(java)学习笔记65:线程的生命周期

    1. 我们学习线程本质就是学习如何开始线程和终止线程.下面这个关于线程的生命周期图,要牢记: 新建状态:当程序使用new关键字创建了一个线程之后,该线程就处于新建状态.此时和其他Java对象一样,它仅 ...