热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

常用的7种排序算法——以leetcode75.SortColors为例

前面:链表的排序参考:链表的排序-冒泡,简单选择,插入排序,归并,快排——以leetcode14

前面:链表的排序参考:链表的排序-冒泡,简单选择,插入排序,归并,快排——以leetcode148. Sort List为例子本文讲数组的排序。

一、题目

Given an array with n objects colored red, white or blue, sort them in-place so that objects of the same color are adjacent, with the colors in the order red, white and blue.

Here, we will use the integers 0, 1, and 2 to represent the color red, white, and blue respectively.

Note: You are not suppose to use the library's sort function for this problem.

Example:

Input: [2,0,2,1,1,0]
Output: [0,0,1,1,2,2]

Follow up:

  • A rather straight forward solution is a two-pass algorithm using counting sort.
    First, iterate the array counting number of 0's, 1's, and 2's, then overwrite array with total number of 0's, then 1's and followed by 2's.
  • Could you come up with a one-pass algorithm using only constant space?

二、解答

本题目实际上考察的而是排序问题,这里总结了7种排序算法的C++经典写法。7种排序算法分别是冒泡,简单选择,直接插入,希尔排序,堆排序、归并排序、快排。

分别的复杂度,及稳定性为下表所示

排序算法复杂度总结
排序算法平均情况最好情况最差情况辅助空间稳定性
冒泡O(n2)O(n)O(n2)O(1)稳定
简单选择O(n2)O(n2)O(n2)O(1)稳定
直接插入O(n2)O(n)O(n2)O(1)稳定
希尔排序O(nlogn)~O(n2)o(n1.3)O(n2)O(1)稳定
堆排序O(nlogn)O(nlogn)O(nlogn)O(1)稳定
归并O(nlogn)O(nlogn)O(nlogn)O(n)稳定
快排O(nlogn)O(nlogn)O(n2)O(logn)~O(n)稳定

稳定性记忆方法:一起(希)快(快排)选(简单选择)排队(堆)

下面是各种排序的代码

class Solution {
public:void sortColors(vector& nums) {//method 1; Bubble sort O(n2) 最好O(n)最差O(n2) beat 22%
// if(nums.empty())
// return;
// int n = nums.size();
// bool flag = true;// for(int i = 0; i != n && flag; ++i)
// {
// flag = false;
// for(int j = n - 2; j >= i; --j)//从后面开始循环
// {
// if(nums[j] // {
// swap(nums[j + 1], nums[j]);
// flag = true;
// }// }
// }
// return;//METHOD 2:simple select sort 简单选择排序 找到最小的再进行交换 O(n2) 最好O(n2) 最差O(n2) beat 99%
// if(nums.empty())
// return;
// int n = nums.size();
// for(int i = 0; i != n; ++i)
// {
// int min = i;
// for(int j = i + 1; j != n; ++j)
// {
// if(nums[j] // min = j;
// }
// if(i != min)
// swap(nums[i], nums[min]);
// }// return;//MEHTOD 3: straight insertion sort 直接插入 O(n2) 最好O(n)最差O(n2)beat 100%// if(nums.empty())// return ;// for(int i = 1; i != nums.size(); ++i)// {// if(nums[i] temp; --j)// nums[j + 1] = nums[j];// nums[j + 1] = temp;//切记此时的时候是nums[j+ 1]没有用了,要覆盖也是覆盖j+ 1// }// }// return ;//method 4: shell sort O(nlogn) - O(n2) 最好O(n1.3)最差O(n2) beat 100%
// if(nums.empty())
// return;
// int n = nums.size();
// int increment = n;
// do
// {
// increment = increment / 3 + 1;
// for(int i = increment; i // {
// if(nums[i] // {
// int temp = nums[i];
// int j = i - increment;//初始化
// for(j = i - increment; j >= 0 && nums[j] > temp; j -= increment)
// nums[j + increment] = nums[j];
// nums[j+ increment] = temp;
// }
// }
// }
// while(increment > 1);// return;//Method 5: Heap sort 三个都是O(nlogn)//简单选择的扩展,每次记录比较的结果 升序一般是大顶堆, 降序一般是小顶堆 堆一定是完全二叉树
// if(nums.empty())
// return;
// int n = nums.size();
// // create
// for(int i = n / 2 - 1; i >= 0; --i)//最大的具有子孩子的节点是n/2-1
// {
// adjustHeap(nums, i, n);
// }// //adjust
// for(int i = n - 1; i >= 0; --i)
// {
// swap(nums[0], nums[i]);//将堆顶元素与末尾元素交换
// adjustHeap(nums, 0, i);//调整剩下的i长度的元素为大顶堆
// }// return; //METHOD 6: merge sort 三个都是O(nlogn) 辅助空间是O(n)稳定// if(nums.empty())// return;// int n = nums.size();// vector res(n);//辅助空间// merge_sort(nums, res, 0, n -1);// return;//METHOD 7: quick sort 最好 平均O(nlogn) 最坏是O(n2)if(nums.empty())return;int n = nums.size();quick_sort(nums, 0, n-1);return;}//the subfunction of Heap sort: adjust heap function
// void adjustHeap(vector & nums, int i, int len)
// {
// int temp = nums[i];
// for(int k = 2 * i + 1; k // {
// if(k + 1 // ++k;
// if(nums[k] > temp)//如果子节点大于父节点,将子节点值赋给父节点(不用进行交换)
// {
// nums[i] = nums[k];
// i = k; //i记录了每次的最大值 节点的索引, 后面只需要交换一次,将最小值temp给索引i的节点
// }// }
// nums[i] = temp;//直交换一次,最小值赋值给上面原最大值 节点的索引处
// return;
// } //the subfunction 1 of merge sort: divide-agency
// void merge_sort(vector & nums, vector & res, int left, int right)
// {
// if(left // {
// int mid = left + (right - left) /2;
// //divide
// merge_sort(nums, res, left, mid);
// merge_sort(nums, res, mid + 1, right);
// //agency
// merge(nums, res, left, right, mid);
// }// }//the subfunction 2 of merge sort: merge sort function
// void merge(vector & nums, vector & res, int left, int right, int mid)
// {
// int i = left;//左半边序列的起始元素索引[left, mid]
// int j = mid+ 1;//右半边序列的起始元素索引[mid+1, right]
// int t = 0;//中间变量,每次t从0开始存储
// while(i <&#61; mid && j <&#61; right)
// {
// res[t&#43;&#43;] &#61; (nums[i] <&#61; nums[j]) ? nums[i&#43;&#43;] : nums[j&#43;&#43;];
// }
// while(i <&#61; mid || j <&#61; right)
// {
// res[t&#43;&#43;] &#61; (i <&#61; mid) ? nums[i&#43;&#43;] : nums[j&#43;&#43;];//注意i &#61;&#61; mid 说明nums[i]用完了&#xff0c;此时应该赋值nums[j&#43;&#43;]
// }
// //将排好序的序列赋回原值
// t &#61; 0;
// while(left <&#61; right)//left递增到right停止
// {
// nums[left&#43;&#43;] &#61; res[t&#43;&#43;];//中间变量,每次t从0开始赋值给left
// }
// return;// }//the subfunction of quick sort: quick sortvoid quick_sort(vector & nums, int left, int right){int pivot;if(left & nums, int left, int right){//int mid &#61; left &#43; (right - left) / 2;int pivot &#61; nums[left];while(left !&#61; right){while(left !&#61; right && nums[right] >&#61; pivot)--right;nums[left] &#61; nums[right];while(left !&#61; right && nums[left] <&#61; pivot)&#43;&#43;left;nums[right] &#61; nums[left];}nums[left] &#61; pivot;return left;}
};



推荐阅读
  • DNN Community 和 Professional 版本的主要差异
    本文详细解析了 DotNetNuke (DNN) 的两种主要版本:Community 和 Professional。通过对比两者的功能和附加组件,帮助用户选择最适合其需求的版本。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本文深入探讨了Linux系统中网卡绑定(bonding)的七种工作模式。网卡绑定技术通过将多个物理网卡组合成一个逻辑网卡,实现网络冗余、带宽聚合和负载均衡,在生产环境中广泛应用。文章详细介绍了每种模式的特点、适用场景及配置方法。 ... [详细]
  • 面临考试压力,急需解决四个编程问题,包括实现乒乓球的动态效果、计算特定日期是一年的第几天、逆序输出数字以及创建弹出菜单。每个问题的解决都能在TC3.0环境中获得50分。 ... [详细]
  • golang常用库:配置文件解析库/管理工具viper使用
    golang常用库:配置文件解析库管理工具-viper使用-一、viper简介viper配置管理解析库,是由大神SteveFrancia开发,他在google领导着golang的 ... [详细]
  • 深入解析JVM垃圾收集器
    本文基于《深入理解Java虚拟机:JVM高级特性与最佳实践》第二版,详细探讨了JVM中不同类型的垃圾收集器及其工作原理。通过介绍各种垃圾收集器的特性和应用场景,帮助读者更好地理解和优化JVM内存管理。 ... [详细]
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • IneedtofocusTextCellsonebyoneviaabuttonclick.ItriedlistView.ScrollTo.我需要通过点击按钮逐个关注Tex ... [详细]
  • 机器学习中的相似度度量与模型优化
    本文探讨了机器学习中常见的相似度度量方法,包括余弦相似度、欧氏距离和马氏距离,并详细介绍了如何通过选择合适的模型复杂度和正则化来提高模型的泛化能力。此外,文章还涵盖了模型评估的各种方法和指标,以及不同分类器的工作原理和应用场景。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • C# WPF 打字射击游戏开发
    介绍了一个基于C#和WPF技术的简单打字射击游戏的实现方法,包括字母的生成、移动、消除以及基本的游戏界面设计。 ... [详细]
  • [转] JavaScript中in操作符(for..in)、Object.keys()和Object.getOwnPropertyNames()的区别
    ECMAScript将对象的属性分为两种:数据属性和访问器属性。每一种属性内部都有一些特性,这里我们只关注对象属性的[[Enumerable]]特征,它表示是否通过for-in循环 ... [详细]
  • 经过一段时间的学习与实践,我已经使用D3.js完成了一些项目。鉴于中文D3教程稀缺,而英文资料虽丰富却对英语水平有一定要求,特此撰写一系列D3实战文章,旨在通过具体案例(如统计数据可视化、地图信息展示等)分享D3的使用技巧,促进技术交流。 ... [详细]
author-avatar
HIGO
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有