我要好offer之 排序算法大总结
1. 插入排序
(1) 直接插入排序
void StraightInsertionSort(std::vector<int>& num) {
    if (num.size() ==  || num.size() == ) return;
    for (int i = ; i < num.size(); ++i) {
        int tmp = num.at(i);
        int j = i - ;
        for (; j >=  && num.at(j) > tmp; --j) {
            num.at(j + ) = num.at(j);
        }
        num.at(j + ) = tmp;
    }
}
(2) 折半插入排序
void BinaryInsertionSort(std::vector<int>& num) {
    if (num.size() ==  || num.size() == ) return;
    for (int i = ; i < num.size(); ++i) {
        int tmp = num.at(i);
        int first = ;   // 最终的first值即为插入位置
        int last = i - ;
        while (first <= last) {
            int mid = first + (last - first) / ;
            if (num.at(mid) < tmp) {
                first = mid + ;
            } else if (num.at(mid) > tmp) {
                last = mid - ;
            } else {
                first = mid;
                break;
            }
        }
        for (int t = i - ; t >= first; --t) {
            num.at(t + ) = num.at(t);
        }
        num.at(first) = tmp;
    }
}
2. 希尔排序
void ShellSort(std::vector<int>& num) {
    if (num.size() ==  || num.size() == ) return;
    for (int gap = num.size() / ; gap > ; gap /= ) {
        for (int i = gap; i < num.size(); ++i) {
            for (int j = i - gap; j >=  && num.at(j) > num.at(j + gap); j -= gap) {
                std::swap(num.at(j), num.at(j + gap));
            }
        }
    }
}
3. 冒泡排序
void BubbleSort(std::vector<int>& num) {
    if (num.size() ==  || num.size() == ) return;
    for (int i = ; i < num.size(); ++i) {
        for (int j = ; j < num.size() - i - ; ++j) {
            if (num.at(j) > num.at(j + )) {
                std::swap(num.at(j), num.at(j + ));
            }
        }
    }
}
4. 快速排序
(1) 递归版
// 划分
int Partition(std::vector<int>& num, int first, int last) {
assert(first >= && first < num.size() && last >= && last < num.size() && first <= last);
int mid = first + (last - first) / ;
std::swap(num.at(first), num.at(mid));
int pos = first;
for (int i = first; i <= last; ++i) {
if (num.at(i) < num.at(first)) {
std::swap(num.at(++pos), num.at(i));
}
}
std::swap(num.at(first), num.at(pos));
return pos;
} // 快速排序
void quickSort(std::vector<int>& num, int first, int last) {
if (first < last) {
int pivotPos = Partition(num, first, last);
quickSort(num, first, pivotPos - );
quickSort(num, pivotPos + , last);
}
} void QuickSort(std::vector<int>& num) {
quickSort(num, , num.size() - );
}
(2) 非递归版
int Partition(std::vector<int>& num, int first, int last) {
    assert(first >=  && first < num.size() && last >=  && last < num.size() && first <= last);
    int mid = first + (last - first) / ;
    std::swap(num.at(first), num.at(mid));
    int pos = first;
    for (int i = first; i <= last; ++i) {
        if (num.at(i) < num.at(first)) {
            std::swap(num.at(++pos), num.at(i));
        }
    }
    std::swap(num.at(first), num.at(pos));
    return pos;
}
void QuickSort(std::vector<int>& num, int first, int last) {
    if (first < last) {
        std::stack<int> stk;
        int pivotPos = Partition(num, first, last);
        if (first < pivotPos - ) {
            stk.push(first);
            stk.push(pivotPos - );
        }
        if (last > pivotPos + ) {
            stk.push(pivotPos + );
            stk.push(last);
        }
        while (!stk.empty()) {
            int right = stk.top();
            stk.pop();
            int left = stk.top();
            stk.pop();
            pivotPos = Partition(num, left, right);
            if (left < pivotPos - ) {
                stk.push(left);
                stk.push(pivotPos - );
            }
            if (right > pivotPos + ) {
                stk.push(pivotPos + );
                stk.push(right);
            }
        }
    }
}
5. 简单选择排序
void SimpleSelectSort(std::vector<int>& num) {
    if (num.size() ==  || num.size() == ) return;
    for (int i = ; i < num.size(); ++i) {
        for (int j = i + ; j < num.size(); ++j) {
            if (num.at(j) < num.at(i)) {
                std::swap(num.at(i), num.at(j));
            }
        }
    }
}
6. 堆排序
// 堆调整,注意结点编号kth从1开始
void HeapAdjust(std::vector<int>& num, int kth, int vecSize) {
int left = * kth;
int right = * kth + ;
int largest = INT_MIN;
if (left <= vecSize && num.at(left - ) > num.at(kth - )) {
largest = left;
} else {
largest = kth;
} if (right <= vecSize && num.at(right - ) > num.at(largest - )) {
largest = right;
}
if (largest != kth) {
std::swap(num.at(kth - ), num.at(largest - ));
HeapAdjust(num, largest, vecSize);
}
} // 建堆
void BuildHeap(std::vector<int>& num) {
for (int i = num.size() / ; i >= ; --i) {
HeapAdjust(num, i + , num.size());
}
} // 堆排序
void HeapSort(std::vector<int>& num) {
BuildHeap(num);
int vecSize = num.size();
while (vecSize > ) {
std::swap(num.at(), num.at(vecSize - ));
--vecSize;
HeapAdjust(num, , vecSize);
}
}
7. 归并排序
void merge(std::vector<int>& num, int first, int mid, int last) {
    std::vector<int> tmp(last - first + , );
    int i = first;
    int j = mid + ;
    int idx = ;
    while (i <= mid && j <= last) {
        if (num.at(i) <= num.at(j)) {
            tmp.at(idx++) = num.at(i++);
        } else {
            tmp.at(idx++) = num.at(j++);
        }
    }
    while (i <= mid) {
        tmp.at(idx++) = num.at(i++);
    }
    while (j <= last) {
        tmp.at(idx++) = num.at(j++);
    }
    for (int i = first; i <= last; ++i) {
        num.at(i) = tmp.at(i - first);
    }
}
void MergeSort(std::vector<int>& num, int first, int last) {
    if (first < last) {
        int mid = first + (last -first) / ;
        MergeSort(num, first, mid);
        MergeSort(num, mid + , last);
        merge(num, first, mid, last);
    }
}
8. 计数排序
void CountSort(std::vector<int>& num) {
    if (num.size() ==  || num.size() == ) return;
    int minVal = *(std::min_element(num.begin(), num.end()));
    int maxVal = *(std::max_element(num.begin(), num.end()));
    int valRange = maxVal - minVal + ;
    std::vector<int> count(valRange, );
    for (int i = ; i < num.size(); ++i) {
        count.at(num.at(i) - minVal)++;
    }
    for (int i = ; i < count.size(); ++i) {
        count.at(i) = count.at(i) + count.at(i - );
    }
    std::vector<int> tmp(num);
    for (int i = tmp.size() - ; i >= ; --i) {
        int lessNum = count.at(tmp.at(i) - minVal);
        num.at(lessNum - ) = tmp.at(i);
        count.at(tmp.at(i) - minVal)--;
    }
}
小结:

我要好offer之 排序算法大总结的更多相关文章
- 剑指offer 查找和排序的基本操作:查找排序算法大集合
		
重点 查找算法着重掌握:顺序查找.二分查找.哈希表查找.二叉排序树查找. 排序算法着重掌握:冒泡排序.插入排序.归并排序.快速排序. 顺序查找 算法说明 顺序查找适合于存储结构为顺序存储或链接存储的线 ...
 - 排序算法大汇总  Java实现
		
一.插入类算法 排序算法的稳定性:两个大小相等的元素排序前后的相对位置不变.{31,32,2} 排序后{2,31,32},则称排序算法稳定 通用类: public class Common { pub ...
 - 我要好offer之 系统基础大总结
		
1. APUE Unix环境高级编程 (1) Unix基础知识: 内核->系统调用->shell和库函数->应用软件 (2) 文件I/O:read函数返回值.进程的文件描述符表.文件 ...
 - 我要好offer之 概率题大总结
		
1. 利用等概率Rand5生成等概率Rand3 Rand5生成等概率Rand3 这个题目可以扩展为:利用等概率RandM生成等概率RandN (M > N) 这里,我们首先明白一个简单的知识点: ...
 - 排序算法大荟萃——希尔(Shell)排序算法
		
1.基本思想:先取一个小于n的整数d1作为第一个增量,把文件的全部记录分成d1个组.所有距离为d1的倍数的记录放在同一个组中.先再各族中进行直接插入排序,然后取第二个增量d2<d1重复上述的分组 ...
 - 我要好offer之 字符串相关大总结
		
1. str*系列手写代码 a. 一定要注意末尾'\0'的处理,切记切记 b. 一定要对输入做有效性判断,多用断言就是了 int Strlen(const char* str) { assert(st ...
 - C语言8大经典排序算法(2)
		
二.插入类排序 插入排序(Insertion Sort)的基本思想是:每次将一个待排序的记录,按其关键字大小插入到前面已经排好序的子文件中的适当位置,直到全部记录插入完成为止. 插入排序一般意义上有两 ...
 - 一文搞定十大经典排序算法(Java实现)
		
本文总结十大经典排序算法及变形,并提供Java实现. 参考文章: 十大经典排序算法总结(Java语言实现) 快速排序算法—左右指针法,挖坑法,前后指针法,递归和非递归 快速排序及优化(三路划分等) 一 ...
 - 十大经典排序算法总结(JavaScript描述)
		
前言 读者自行尝试可以想看源码戳这,博主在github建了个库,读者可以Clone下来本地尝试.此博文配合源码体验更棒哦~~~ 个人博客:Damonare的个人博客 原文地址:十大经典算法总结 这世界 ...
 
随机推荐
- python之道08
			
1.有如下文件,a1.txt,里面的内容为: 某某是最好的学校, 全心全意为学生服务, 只为学生未来,不为牟利. 我说的都是真的.哈哈 分别完成以下的功能: a,将原文件全部读出来并打印. 答案 f ...
 - Dojo的dojoConfig函数
			
在我们引入 Dojo 的时候都会先做一些全局的配置,所使用的就是 Dojo 的 Config 接口. dojoConfig为以前的dgConfig函数. <script type="t ...
 - SQL Server 游标的应用
			
----------------SQL游标应用----------------- 今天由于业务需求,需要在存储过程中实现有一个表的主键去匹配在另一个表中作为外键所对应的数值 ,若在C#中则非常简单只需 ...
 - Element-ui tree组件自定义节点使用方法
			
工作上使用到element-ui tree 组件,主要功能是要实现节点拖拽和置顶,通过自定义内容方法(render-content)渲染树代码如下~ <template> <di ...
 - MFC里 显示设备上下文CClient dc(this) 和 CPaintDC dc(this)
			
1 CPaintDC类(1)CPaintDC类是CDC类的一个派生类,该类一般用在响应WM_PAINT消息的函数OnPaint()中.(2)WM_PAINT消息是当窗口的某个区域需要重画时激发的窗口消 ...
 - Redis 和缓存技术
			
Redis 是什么?什么作用?优点和缺点? https://blog.csdn.net/weixin_42295141/article/details/81380633 Redis 的主要功能哨兵+复 ...
 - rest_framework之status HTTP状态码
			
Django Rest Framework有一个status.py的文件 通常在我们Django视图(views)中,HTTP状态码使用的是纯数字,像400,404,200,304等,并不是那么很好理 ...
 - linux下的一些命令分析与shell的一些命令
			
对> 与 >>的理解 echo "aaa" > aaa.txt 这个是在aaa.txt中写入aaa 可以用cat aaa.txt查看 echo &qu ...
 - MySQL练习50题
			
介绍一个学习SQL的网站:https://sqlbolt.com/ 习题来源于网络,SQL语句是自己的练习答案,部分参考了网络上的答案. 花了一晚上的时间做完,个人认为其中的难点有:分组提取前几名的数 ...
 - 经典MSSQL语句大全和常用SQL语句命令的作用
			
下列语句部分是Mssql语句,不可以在access中使用. SQL分类: DDL类型包括数据库.表的创建,修改,删除,声明—数据定义语言(CREATE,ALTER,DROP,DECLARE) DML类 ...