# 排序 ## 常考排序 ### 快速排序 ```c void QuickSort(vector& nums, int n){ // 思路:把一个数组分为左右两段,左段小于右段 quickSort(nums, 0, n - 1); } // 原地交换,所以传入交换索引 void quickSort(vector& nums, int start, int end) { if (start < end) { // 分治法:divide int pivot = partition(nums, start, end); quickSort(nums, 0, pivot-1); quickSort(nums, pivot+1, end); } } // 分区 int partition(vector& nums, int start, int end){ // 选取最后一个元素作为基准pivot int p = nums[end]; int i = start; // 最后一个值就是基准所以不用比较 for (int j = start; j < end; j++) { if (nums[j] < p) { swap(nums[i], nums[j]); i++; } } // 把基准值换到中间 swap(nums[i], nums[end]); return i; } ``` ### 归并排序 ```c void MergeSort(vector& nums){ if(nums.size() <= 1) return; mergeSort(nums, 0, nums.size() - 1); } void mergeSort(vector& nums, int start, int end){ if (start < end) { // 分治法:divide 分为两段 int mid = (start + end) / 2; mergeSort(nums, start, mid); mergeSort(nums, mid + 1, end); // 合并两段数据 merge(nums, start, mid, end); } } void merge(vector& nums, int start, int mid, int end){ vector temp(end - start + 1, 0); int i = start; int j = mid + 1; int k = 0; while (i <= mid && j <= end) { temp[k++] = (nums[i] <= nums[j]) ? nums[i++] : nums[j++]; // 谁小合并谁 } // 剩余部分合并 while (i <= mid) { temp[k++] = nums[i++]; } while (j <= end) { temp[k++] = nums[j++]; } for (i = 0; i < k; i++) { nums[start + i] = temp[i]; } } ``` ### 堆排序 用数组表示的完美二叉树 complete binary tree > 完美二叉树 VS 其他二叉树 ![image.png](https://img.fuiboom.com/img/tree_type.png) [动画展示](https://www.bilibili.com/video/av18980178/) ![image.png](https://img.fuiboom.com/img/heap.png) 核心代码 ```c //堆排序 void heapSort(vector& nums){ // 1、无序数组nums // 2、将无序数组nums构建为一个大根堆 buildMaxHeap(nums); // 3、交换nums[0]和nums[nums.size()-1] // 4、然后把前面这段数组继续下沉保持堆结构,如此循环即可 int len = nums.size(); //此处len需要变换,单独拎出来 for (int i = nums.size() - 1; i > 0; i--) { swap(nums[0], nums[i]); // 从后往前填充值 len--; // 前面的长度也减一 heapHelper(nums, 0, len); } } // 建立大顶堆 void buildMaxHeap(vector& nums){ int length = nums.size(); for (int i = length / 2; i >= 0; i--) { heapHelper(nums, i, length); } } //堆调整 void heapHelper(vector& nums, int i, int length){ int left = 2 * i + 1; // 左节点索引(从0开始,所以左节点为i*2+1) int right = 2 * i + 2; // 右节点索引 int largest = i; // largest保存根、左、右三者之间较大值的索引 // 存在左节点,左节点值较大,则取左节点 if (left < length && nums[left] > nums[largest]) { largest = left; } // 存在有节点,且值较大,取右节点 if (right < length && nums[right] > nums[largest]) { largest = right; } // 如果非根节点较大,则继续递归 if (largest != i) { swap(nums[i], nums[largest]); heapHelper(nums, largest, length); } } ``` ## 参考 [十大经典排序](https://www.cnblogs.com/onepixel/p/7674659.html) [二叉堆](https://labuladong.gitbook.io/algo/shu-ju-jie-gou-xi-lie/er-cha-dui-xiang-jie-shi-xian-you-xian-ji-dui-lie) ## 练习 - [ ] 手写快排、归并、堆排序